Angular JS is probably the best javascript framework around. Simple things are simple and complicated things are possible. As with any other framework in any language, one has to adapt the thought patterns around the intricacies and the peculiarities of the language and the library. One has to take time to learn Angular, but once it is finally understood, the productivity is enhanced.
I have done some web pages with angular. For one I wanted a timer object that I coud stop, start, set the period and the method at will. For instance, within a controller or a directive or any other Angular object:
var timer = new Timer(5000, myFunction, $timeout).start()
This will start the timer and run myFunction every five seconds.
timer.delay(3);
This function will delay the firing of the timer for 3 periods (in the case, 15 seconds.
The rest of the functions should be obvious.
The implementation
The code should be simple to just copy and paste. It uses the $timeout global object from Angular, som make sure angular.js is imported into the page.
var Timer = function(period, meth, $timeout) {
var tID = null;
var tPeriod = period || 5000;
var tDelay = 0;
var tMeth = meth || null;
function delay(periods) {
tDelay = periods;
return this;
}
function setPeriod(period) {
tPeriod = period || 5000;
return this;
}
function setMethod(meth) {
tMeth = meth || null;
return this;
}
function stop () {
if (typeof(tID) != null) {
$timeout.cancel(tID);
tID = null;
}
return this;
}
function start() {
tID = $timeout(function () {
if (tDelay) {
tDelay -= 1;
} else {
if (tMeth) {
tMeth();
}
}
start();
}, tPeriod);
return this;
}
function restart() {
stop();
start();
return this;
}
return {
'delay': delay,
'restart': restart,
'setMethod': setMethod,
'setPeriod': setPeriod,
'start': start,
'stop': stop
}
}
I have always been bothered with the session files left by emacs is user-emacs-directory, normally ~/.emacs.d/. These files have the name session.[a long string of hexadecimal characters] and save the session variables when emacs exits without being explicitely terminated. Like when one ends up the session by the session manager and does not bother to close the emacs window first.
After a few days, a set of session files ends up cluttering the emacs directory. These files lose their purpose once one does not want to restore the session on the next emacs invocation, but they are not purged automatically.
The solution
purge-session-files deletes all session files, preserving the n most recent. n defaults to 1. The function can be called interactively. In such case, the user is given the choice of the number of session files to preserve (defaulting to one).
Here is the code:
(defun purge-session-files (&optional n)
"Purge all session files but the n more recent files."
(interactive (list (read-number "Number of files to preserve" 1)))
(let ((files (directory-files user-emacs-directory t "session.*")))
(dolist (file (butlast files (or n 1)))
(delete-file file nil))))
Some time ago, I wrote a XDG Directory Specification for GNU Emacs. This is used to assign directories for emacs files under the XDG specification for the home directory.
The named directories are set from the environment variables mentioned in the standard. If no environment variables are present, the following defaults are used:
Symbol
Environment variable (+ /emacs)
Default directory
user-emacs-data-directory
XDG_DATA_HOME
~/.local/share/emacs/
user-emacs-config-directory
XDG_CONFIG_HOME
~/.config/emacs/
user-emacs-cache-directory
XDG_CACHE_HOME
~/.cache/emacs/
user-emacs-lisp-directory
`user-emacs-data-directory`/lisp/
user-documents-directory
output from xdg-user-dir, if such executable exists
The following files are my configuration for gEDA. gEDA is a suite for electronic design automation (EDA, unsurprisingly). It is a set of loose tools that are supposed to work together, even if with rough edges, to provide an end-to-end solution to design electronic boards.
It actually does a good job. Most of the complaints on gschem (the schematic editor) are not to blame on the program, but on the symbol libraries, which, despite the good work done there, are ridden with annoying inconsistencies. Unlike any commercial applications with free versions, there are no limits on pin number or board size. Once you get the hang of it, it should be enough for most small or medium hobbyist project.
gEDA tools are configured with Guile. Guile is an implementation of scheme. The language is, I grant it, not the best choice for a scientific or technical program, given the profusion of mathematical libraries in python (scipy, numpy, etc). But it does the job and does it well.
What this file does is the following: a set of subdirectories are assigned (if they exist) for my own symbols, subcircuits and scripts, in ~/.gEDA/. There will be directories under the directory gschem was started from, normally the project directory. As these configurations are quite common, I have moved them here. I have found them to be quite comprehensive. I do not usually have per directory configuration files, since they are not needed.
I do not fancy the large symbols in gschem libraries. It wastes a lot of paper for nothing. As I cannot reduce the symbols without implementing the whole library, I chose to widen the paper, that is, to enlarge the titleblock and the borders of the paper. Once the schematic is printed, it will be printed under the new units, since I just print it complete without margins on A3 paper.
The symbol files can be found at this Google Drive directory. There is a symbol for a title block (with useful attributes), A3 landscape and A4 portrait borders. The following image is a screenshot of gschem running under Linux with the title block and borders I have provided.
Haskell is not an easy language. It takes a lot of mind wrapping to understand monads, but once you do, you lose all capability to explain it. I shall not even try to explain monads. I prefer to show by example how haskell should be written.
You can, simplistically, say a monad is data with a context. Just like with physical units, like 2m or 2Kg. Two metres are different from two kilogrammes. One pertains to length, another to mass. In monadic speak, it would be "Metre 2" or "Kilogramme 4", since monads are prefixed. You could add metres with metres, but not metres with kilogrammes. And thus type safety comes into being actually useful.
The following code is here. It has been edited in Literate Haskell and transformed with Pandoc to HTML. The original HTML file, as it came out of Pandoc, can be found here.
A Date/Time Period
DESCRIPTION: A period that contains dates or times.
AUTHOR: Francisco Miguel Colaço <>
DATE: 2013-12-23
Summary
The Period class describes a possibly infinite period of dates and times. It has functions that assess if a date or a moment in time belongs in a given periods. It has also fuctions that create new intervals with reference to the current moment (under the IO Monad).
Implementation
Declarations
This package uses several Haskell language extensions. These extensions are not deprecated, nor considered harmful.
{-# LANGUAGE DeriveDataTypeable #-}{-# LANGUAGE GADTs #-}{-# LANGUAGE OverloadedStrings #-}{-# LANGUAGE StandaloneDeriving #-}
A period can extend From a start date or time, Until an end time, with no beginning, Between two times (or dates), or a bottom value of Infinite, that is, from the start to the end of times. To be precise, one could make periods out of integers or numbers. The only requisite is that the type variable is orderable and showable. All numbers are. Of course, one could restrict the period type variable to dates and times only by having them belong to the Data.Time.Format.FormatTime class. We opted not to.
Using GADTs, the type constraints of the type variable are made right at the constructors, and thus reflect themselves through all the functions that use the Period data type.
-- | An interval of orderable and showable values. The period can be `From` a-- determined time with no end; `Until` a given time and with no start;-- `Between` two times; or `Infinite`, meaning without start or end.dataPeriod a whereFrom :: (Ord a, Show a) => a ->Period a
Until :: (Ord a, Show a) => a ->Period a
Between :: (Ord a, Show a) => a -> a ->Period a
Infinite ::Period a
instanceShow (Period a) whereshow (From start) ="From "++show start
show (Until end) ="Until "++show end
show (Between start end) ="Between "++show start ++" and "++show end
showInfinite="All times"
derivinginstanceTypeable1Periodderivinginstance (Data a, Ord a, Show a) =>Data (Period a)
A DatePeriod is a period of dates (days). A TimePeriod is a period of UTC times. Both values are defined in Data.Time.
Periods can also be set in terms of any orderable and showable types, like enumerations or numbers. The library is to be used with dates and times.
Conversions between the two period types are possible.
-- | Converts from a time period to a date period. Truncates the period to the-- dates, disregarding the day fractions.toDatePeriod ::TimePeriod->DatePeriod
toDatePeriod (From start) =From (utctDay start)
toDatePeriod (Until end) =Until (utctDay end)
toDatePeriod (Between start end) =Between (utctDay start) (utctDay end)
toDatePeriod Infinite=Infinite-- | Converts from a date period to a time period. The period is counted from-- the start of the first day to the end of the last.toTimePeriod ::DatePeriod->TimePeriod
toTimePeriod (From start) =From (UTCTime start 0.0)
toTimePeriod (Until end) =Until (UTCTime end 86399.999999)
toTimePeriod (Between start end) =Between (UTCTime start 0.0) (UTCTime end 86399.999999)
toTimePeriod Infinite=Infinite
Period Limit Tests
A period is closed when it has no upper end. That is, only Infinite and From periods are open. All others are closed.
-- | Tells if the period has an end.isClosed ::Period a ->Bool
isClosed Infinite=False
isClosed (From _) =False
isClosed _ =True
Any period may or may not have have a start or an end. As it may have not, the enquiry can fail. So, the answer is wrapped in the Maybe monad.
From periods have a start, but not an end. Until periods have only an end. Between periods have both. Infinite has neither.
-- | Returns the start of the period (in the Maybe Monad), if it has one.periodStart ::Period a ->Maybe a
periodStart (From start) =Just start
periodStart (Between start _) =Just start
periodStart _ =Nothing-- | Returns the end of the period (in the Maybe Monad), if it has one.periodEnd ::Period a ->Maybe a
periodEnd (Until end) =Just end
periodEnd (Between _ end) =Just end
periodEnd _ =Nothing
Value Containment
contains is meant to be used infix. Tells if a given period contains the given date. A period has inclusive bounds. A date exactly equal to one of the bounds is still contained in it.
contains ::Period a -> a ->Bool
contains Infinite _ =True
contains (From start) now = now >= start
contains (Until end) now = now <= end
contains (Between start end) now = (now >= start) && (now <= end)
belongs is contains with the arguments splitted. One date belongs to a period when such period contains the date.
belongs :: a ->Period a ->Bool
n `belongs` p = p `contains` n
Constructors Reported to the Current Date
Constructors are provided that extend from or until the current day and time. The constructors are computed under the IO Monad, since they have to compute the current date.
-- | Returns a time period that extends from the present time.fromNow ::IOTimePeriod
fromNow = liftM From getCurrentTime
-- | Returns a time period that extends until the present time.untilNow ::IOTimePeriod
untilNow = liftM Until getCurrentTime
-- | Returns a day period that extends from the present day.fromToday ::IODatePeriod
fromToday = liftM (From. utctDay) getCurrentTime
-- | Returns a day period that extends until the present day.untilToday ::IODatePeriod
untilToday = liftM (Until. utctDay) getCurrentTime
Common Date Periods
Date periods for the current year, the current month and the current week are often requested. These have to be computed under the IO Monad, since they determine the current time.
-- | Returns a date period that spans throughout the current year.thisYear ::IODatePeriod
thisYear =do
(yr, _, _) <- (liftM $ toGregorian . utctDay) getCurrentTime
return$Between (fromGregorian yr 11) (fromGregorian yr 1231)
-- | Returns a date period that spans throughout the current month.thisMonth ::IODatePeriod
thisMonth =do
(yr, mo, _) <- (liftM $ toGregorian . utctDay) getCurrentTime
return$Between (fromGregorian yr mo 1) (fromGregorian yr mo $ gregorianMonthLength yr mo)
-- | Returns a date period that spans throughout the week.thisWeek ::IODatePeriod
thisWeek =do
(yr, wk, d) <- (liftM $ toWeekDate . utctDay) getCurrentTime
return$Between (fromWeekDate yr wk 1) (fromWeekDate yr wk 7)
A time period is quite requested: the one that comprises the current day. This period has to be computed under the IO Monad. The implementation has two flaws: do not account for end days and fall short one microssecond from the real end of the day.
-- | Returns a time period that comprises the current day.thisDay ::IOTimePeriod
thisDay =do
today <- liftM utctDay getCurrentTime
return$ toTimePeriod $Between today today
A very frequent query is to know if a given period contains the current day or time. These queries have to be made inside the IO Monad.
-- | Tells if a date period contains the current day.containsToday ::DatePeriod->IOBool
containsToday period =do
now <- getCurrentTime
return$ period `contains` utctDay now
-- | Tells if a time period contains the current time.containsNow ::TimePeriod->IOBool
containsNow period =do
now <- getCurrentTime
return$ period `contains` now