This is a Prooshious gunn. This is a ffrinch. Tip.
This is the flag of the Prooshious, the Cap and Soracer.
This is the bullet that byng the flag of the Prooshious.
This is the ffrinch that fire on the Bull that byng the
flag of the Prooshious. Saloos the Crossgunn!
Up with your pike and fork! (Bullsfoot! Fine!)
This is the triplewon hat of the Lipoleum.
Tip. Lipoleumhat.
[Click.]
- Introduction to the Utilities
- Function: repeat
- Function: assign
- Function: isDigit
- Function: indexOfLongestSuffixSansDigits
- Object: CharCode
- Object: KeyCode
- Object: Angle
- Object: Measure
- Object: FunctionCall
Introduction to the Utilities
We begin our deconstruction of Slytherins Wake by examining the program’s utilities. Each utility is small and may be used anywhere in Slytherins Wake. None of the utilities depend on anything except the type of programming environment typically provided by many popular web browsers. We assume that a JavaScript interpreter provided by a modern web browser such as such as Chrome, Edge, or Firefox will be used to execute the program and that the brower’s implementation of JavaScript supports the language features and standard library routines used by Slytherins Wake.
The utilities we’re about to look at won’t shed much light on the anatomy of a snake. However, we should be familiar with them so that when we encounter one of the utilities in the context of another component, like a constructor function or another utility, we'll recognize it, remember what it does, and recall how it should be used if not exactly how it is implemented.
Some of the utilities described below are first-class functions. The others are frozen objects identified by constant variables. First we’ll look at the functions, which are like loose tools strewn about on a programmer’s workbench. Then we’ll examine the objects, which are like toolboxes containing related tools, parts, and paraphernalia.
Go slow. Study each part carefully. Read the notes. Follow the links that follow the notes. Do the exercises.
NOTES
- This tapestry of text and source code that we’ve woven, which we’re now unraveling, is intended for highbrow coders everywhere and for the lowly worms who wrote it. It’s for experts and novices alike. In other words, we’re trying to satisfy everybody at the risk of humoring only him.
- We’ve sequenced the sections of Slytherins Wake such that little things are presented first, for the reader’s sake, thence we proceed to greater. This will appeal to those of you who enjoy a bottoms-up approach to making acquaintances. If you’re on a high harse and like to leer at things from the top down, come down and slog around in the mud with us for a while until we reach a level of abstraction that dispenses with dirty details you prefer not to deal with.
- We try to cater to the diverse needs of our readers by providing links to online resources about topics more or less relevant to Slytherins Wake in the sections where those topics are first introduced. Many of the links refer to pages on the website https://developer.mozilla.org. We’re all in on MDN. It’s our favorite online source of information about JavaScript. Thankfully, it’s add-free, as in cholesterol-free, sugar-free, gluten-free, …
- Besides hyperlinks, we also provide notes like this one. Similar to notes found in a study Bible, our notes are for students who seek a deeper level of understanding than their author, who forgets things easily.
- What’s noteworthy? you ask. That’s for us to decide and you to wonder about. Sometimes we find ourself in a rabbit hole. Take the note that follows this one, for example. It fascinates us, which might make you wonder.
- According to the Wikipedia article https://en.wikipedia.org/wiki/Study_Bible, “Nearly all Catholic Bibles have explanatory and interpretative footnotes.[citation needed] [sic] For example, the Jerusalem Bible is a widely respected study Bible originally made by French monks in Jerusalem, under the auspices of the Catholic Church. The original French edition of 1961 became the basis of versions of this study Bible in several other languages, including English, revised as the New Jerusalem Bible; some versions have more extensive notes than others.”
SEE
- https://developer.mozilla.org/en-US/docs/Web/Tutorials
- https://developer.mozilla.org/en-US/docs/Glossary/JavaScript
- https://developer.mozilla.org/en-US/docs/Glossary/ECMAScript
- https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/Functions
- https://developer.mozilla.org/en-US/docs/Glossary/First-class_Function
- https://developer.mozilla.org/en-US/docs/Learn/JavaScript/Objects/Basics
- https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/const
- https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/freeze
SEE ALSO
- https://en.wikipedia.org/wiki/Tilting_at_windmills
- http://www.laputan.org/mud/mud.html
- https://en.wikipedia.org/wiki/Top-down_and_bottom-up_design
- https://en.wikipedia.org/wiki/Abstraction
- https://www.newyorker.com/culture/cultural-comment/going-all-in-on-all-in
- https://en.wikipedia.org/wiki/Study_Bible
Function: repeat.
We now present
repeat
, a simple utility function designed to execute another function multiple times. The function seems contrived and unnecessary to us. Its author says he hid an agenda somewhere around here, but he can’t remember where he put it. Can you help him find it?/** * Given a function
foo
and a non-negative integern
, *repeat
calls foon
times. * The value returned by the function isundefined
. */ function repeat(foo, n) { while (n > 0) { foo(); n = n - 1; } }NOTES
- This piece of Slyterins Wake can be partitioned into two parts: a block of innocuous comments followed by some JavaScript code that defines a function. Comments that appear before a unit of code, such as the definition of a function or an object, typically describe the publicly accessible components or services the unit provides and how programmers may use it. (Comments have no effect on how a program runs.) Additional commentary might be needed to put a piece of a program in perspective so that readers thoroughly understand the contribution the piece makes to the whole program. Providing readers with contextual information through prose that is less constrained and more readable than tradition comments is a strategy that literate programmers adopt in their quest to help people, not machines, comprehend their code.
- Self-anointed literate programmers express themselves how ever they choose. That’s why their bible is called The Art, not The Dogma of Computer Programming. You can also view the source code for Slytherins Wake—including traditional comments like the comments shown in the code snippet above but without all this hot air—over at Tangled, the Reader’s Digest version of the program. Less is more and then some.
- A value passed to a JavaScript function can be a reference to another function just as easily as it can be a string or a number or any other type of data. This is a noteworthy feature of JavaScript that we’ll take advantage of repeatedly in Slytherins Wake.
- JavaScript variables, like the parameters
foo
andn
, can be assigned values of any type. The functionrepeat
does not verify that the value offoo
is actually of typefunction
or that the value ofn
is in fact a non-negative integer. Therepeat
utility and the other utilities described in this section are only intended to be used within Slytherins Wake; they are not intended to be used by developers who compose new programs using Slytherins Wake. Therefore, we have, we assert, complete control over all values passed to these utility functions and methods; we assume responsibility for the arguments we pass, ane we assure you they are of the correct type and are in range, assuming we haven’t made mistakes to the contrary. - The fixed number of characters needed to define the function aside, every time we invoke
repeat
instead of writing a conventional loop, Slyterins Wake ends up being little smaller than it would have been if we had used a loop instead of calling the function. (Maybe this is that hidden agenda he’s searching for? The function still seems shaky, if you ask us.) - Removing or tampering with
repeat
is verboten, especially if you don’t know what Chesterton’s fence is for. (He insisted we tell you that. Frankly, we’re not sure what all the fuss is about, much less how fences factor in.)
SEE
- https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Lexical_grammar
- https://developer.mozilla.org/en-US/docs/Web/JavaScript/Data_structures
- https://developer.mozilla.org/en-US/docs/Glossary/Parameter
- https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/while
- https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Comparison_Operators#Relational_operators
- https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Arithmetic_Operators#Subtraction_(-)
- https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Assignment_Operators
- https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/undefined
SEE ALSO
- https://en.wikipedia.org/wiki/Wikipedia:Chesterton%27s_fence
- https://en.wikipedia.org/wiki/Foobar
- https://developer.mozilla.org/en-US/docs/Mozilla/Developer_guide/Coding_Style
- https://en.wikipedia.org/wiki/Less_(Unix)
- https://en.wikipedia.org/wiki/More_(command)
EXERCISES
- How often is the function
repeat
invoked in Slytherins Wake? - What is
repeat
’s raison d'ĂȘtre? - Is there a limit to the number of times a JavaScript function can be invoked using
repeat
? If there is a limit, what is it and why does the limit exist? - What are the values of the following JavaScript expressions?
'Doh!' - 1
NaN > 0
- Is the function
repeat
an algorithm? Is it a computational method? Explain. - Would you change
repeat
if you couldn’t be sure that the values assigned to the parametersfoo
andn
were always going to be a reference to a function and a finite number respectively? If you wouldn’t change the function, why not? If you would change the function, why and how would you change it? (If you think the function should be changed, consider using athrow
statement, thetypeof
operator, theisFinite
global function, theNumber.isFinite
method, or anything else that might help.) - Make an interesting change to the way repeat is implemented without changing the name of the function, the number or order of its parameters, or the service that
repeat
purports to provide. Test your change. Make sure the new version ofrepeat
works just like the original version. Repeat as many times as you can or until you get tired of this exercise. Which variation ofrepeat
do you like best? Why? - Enhance
repeat
. Add a third parameter calledargs
. In your enhanced version of the function, invokefoo
by callingfoo.apply
. Pass the value ofargs
to theapply
method. Use your modified version ofrepeat
together with thealert
command to display the salutationHello, Durlbin!
in a popup window twice in a row. - What happens when you execute the following command using the version of repeat that you created in the previous exercise?
repeat(repeat, 2, [alert, 2, ["Finis, again!"]]);
- Try to compose a method using the Java programming language that does essentially the same thing that
repeat
does. Compare and contrast your Java method with the JavaScript function. If you could not create a comparable subroutine using Java, what inhibited you?
Function: assign.
/** * Giving two objects,
target
andsource
, *assign
copies enumerable properties fromsource
totarget
* and returns the value oftarget
. */ function assign(target, source) { for (let propertyName in source) { target[propertyName] = source[propertyName]; } return target; }Note: The
assign
utility is similar to the standard library methodObject.assign
.SEE
- https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/for...in
- https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/let
- https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Property_accessors
- https://developer.mozilla.org/en-US/docs/Web/JavaScript/Enumerability_and_ownership_of_properties
SEE ALSO
Function: isDigit.
/** * Given a string s and an optional number i, *
isDigit
returnstrue
if: * a) the value of i isundefined
ornull
* and the first character of s is a digit, or * b) the character at index i′ of s is a digit where * i′ is the value of i converted to an integer. * Otherwise,isDigit
returnsfalse
. */ function isDigit(s, i) { return CharCode.isDigit(s.charCodeAt(i)); }Note:
- The description of
isDigit
above is ambiguous in at least one way. It doesn’t say how the value ofi
will be converted to an integer if it is not an integer to begin with. Will the value be rounded up or down? What if the value ofi
is a string, not a number? Since the conversion is performed byString.prototype.charCodeAt
and we can’t see how that method does its job, it’s not easy to really know what will happen in all cases. Different web browser vendors may implementString.prototype.charCodeAt
differently, and Slytherins Wake isn’t intended to be tied to just one browser. We can test various suspect scenarios using various versions of various web browsers and discover howcharCodeAt
behaves. But just talking about all those permutations makes our head hurt. What about browsers and input combinations that we don’t test? Is there authoritative documentation somewhere that says howString.prototype.charCodeAt
should work? These are important questions to ponder.
See
- The description of
Function: indexOfLongestSuffixSansDigits.
/** * Returns the index of the beginning of the longest suffix of * string
s
that does not contain any digits. */ function indexOfLongestSuffixSansDigits(s) { let i = s.length - 1; while (i > 0 && !isDigit(s, i)) { i = i - 1; } return i + 1; }Object: CharCode.
const CharCode = Object.freeze({
CharCode.<property>
Zero: '0'.charCodeAt(0), Nine: '9'.charCodeAt(0),
CharCode.isDigit
isDigit: function(code) { return CharCode.Zero <= code && code <= CharCode.Nine; }
});
Object: KeyCode.
const KeyCode = Object.freeze({
KeyCode.<property>
Space: 32, LeftArrow: 37, UpArrow: 38, RightArrow: 39, DownArrow: 40
});
Object: Angle.
const Angle = Object.freeze({
Angle.<properties>
halfTurn: '180deg', straight: '0deg', quarterTurnLeft: '-90deg', quarterTurnRight: '90deg'
});
Object: Measure.
const Measure = Object.freeze({
Measure.indexOfUnits
indexOfUnits: function (string) { return indexOfLongestSuffixSansDigits(string); },
Measure.parseNumber
parseNumber: function (value) { const number = parseFloat(value); return (isFinite(number)) ? number : 0; },
Measure.units
units: function (string) { return string.slice(Measure.indexOfUnits(string)); },
Measure.separate
separate: function (string) { const i = Measure.indexOfUnits(string); return { number: Measure.parseNumber(string.slice(0, i)), units: string.slice(i) }; }
});
Object: FunctionCall
const FunctionCall = Object.freeze({
FunctionCall.arguments
arguments: function (string, fname) { if (fname === undefined) fname = ""; const s = fname + '('; const i = string.indexOf(s); const j = string.indexOf(')', i); return (i < 0 || j < 0) ? "" : string.substring(i + s.length, j); }
});
This way the museyroom. Mind your boots goan out.
[Click.]