--------------------------------------------------------------------------- --GENERAL PURPOSE SPIFFY routines for FM music --------------------------------------------------------------------------- -- Feel free to use any of these routines in your own Euphoria programs. -- All of the routines in this file are general-purpose routines that -- are used by FM.EX but dont do anything FM-specific --------------------------------------------------------------------------- include misc.e include file.e include get.e --------------------------------------------------------------------------- ---constants--- constant true=1 constant false=0 global constant stdin=0 global constant stdout=1 global constant failure=-1 --globals-- global integer PATH_SLASH if platform()=LINUX then --yes, believe it or not, HSPEAK can be compiled to run on Linux PATH_SLASH='/' else PATH_SLASH='\\' end if --redundant to the standard upper() and lower() functions I know, but I --use them constantly, and almost never use the other wildcard functs. --this avoids including wildcard.e -- convert atom or sequence to lower case global function hs_lower(object x) return x + (x >= 'A' and x <= 'Z') * ('a' - 'A') end function -- convert atom or sequence to upper case global function hs_upper(object x) return x - (x >= 'a' and x <= 'z') * ('a' - 'A') end function ---convert a delimited string into a sequence of strings--- global function explode(sequence s,sequence delim) sequence result integer pos result={} pos=match(delim,s) if pos then while pos do result=append(result,s[1..pos-1]) s=s[pos+length(delim)..length(s)] pos=match(delim,s) end while end if result=append(result,s) return result end function --merge a sequence of strings into a single delimited string-- global function implode (sequence s,sequence glue) sequence output output={} if length(s) then output=s[1] for i=2 to length(s) do output &= glue & s[i] end for end if return output end function ---exclude specified chars from a string sequence--- global function exclude(sequence s,sequence c) sequence result result={} for i=1 to length(s) do if not find(s[i],c) then result=append(result,s[i]) end if end for return(result) end function --wraps value() and returns a default if it fails global function string_to_object(sequence s,object default) object result result=value(s) if result[1]!=GET_SUCCESS then result[2]=default end if return result[2] end function ---collapse reundant elements of a sequence--- global function redundx(sequence s,object c) sequence n n="" for i=1 to length(s) do if compare(s[i],c)=0 then if i=start do seq=seq[1..at-1] & new & seq[at+length(old)..length(seq)] start=at+length(new) at=match(old,repeat(0,start-1) & seq[start..length(seq)]) end while return(seq) end function --normalize a pathname to use forward slashes-- global function filenamix(sequence s) return substitute(s,'\\','/') end function --normalize a pathname to use back slashes-- global function filenamos(sequence s) return substitute(s,'/','\\') end function --normalise a pathname to use the platform specific path delimiter-- global function normalize_filename(sequence s) if platform()=LINUX then return filenamix(s) else return filenamos(s) end if end function --extract only portion of a string after the last of a delimiter-- global function get_suffix(sequence s,sequence delim) sequence broken broken=explode(s,delim) return broken[length(broken)] end function --extract all of a string except what follows the last of a delimiter-- global function without_suffix(sequence s,sequence delim) sequence broken broken=explode(s,delim) if length(broken)>1 then return implode(broken[1..length(broken)-1],delim)&delim else return "" end if end function --exclude the path from a fully qualified filename-- global function file_only(sequence filename) return get_suffix(filenamix(filename),"/") end function --return only the path from a fully qualified filename-- global function path_only(sequence filename) return without_suffix(filenamix(filename),"/") end function --return only the extension of a filename-- global function extension_only(sequence filename) return get_suffix(filename,".") end function --exclude the extension from a filename-- global function without_extension(sequence filename) sequence result result=without_suffix(filename,".") if length(result) then return result else return filename&"." end if end function --returns a filename with the extension changed-- global function alter_extension(sequence filename,sequence newext) return without_extension(filename) & newext end function --return the larger of two integers-- global function large(integer n1,integer n2) if n1>n2 then return n1 else return n2 end if end function --return the smaller of two integers-- global function small(integer n1,integer n2) if n1255 then --not a char return 0 end if end for return quad[1]*#1000000 + quad[2]*#10000 + quad[3]*#100 + quad[4] end function --------------------------------------------------------------------------- --converts a sequence of 2 bytes in {2,1} order into a single int global function word_to_integer(sequence word) if length(word)!=2 then --not a real word return 0 end if for i=1 to 2 do if not integer(word[i]) then --not an integer return 0 end if if word[i]<0 or word[i]>255 then --not a word return 0 end if end for return word[1]*#100 + word[2] end function --------------------------------------------------------------------------- global function write_lump(integer filehandle,sequence name,sequence data) if length(name)>12 then --fail if name is too long return(false) end if puts(filehandle,hs_upper(name)&0) puts(filehandle,absurd_byte_order(length(data))) puts(filehandle,data) return(true) end function --------------------------------------------------------------------------- --returns the first x characters, avoiding word break. --this is used to do the word-wrapping of long-lines global function before_wrap_point(sequence string) sequence line,word,result integer size size=80 result=string line=explode(string,"\n") if length(line) then word=explode(line[1]," ") result="" if length(word) then while length(word) do if length(result & word[1]) <= size then result &= word[1] word=decapitate(word) if length(word) then result &= " " end if else if length(word[1])>size then result &= word[1][1..large(0,size-length(result))] end if exit end if end while else result&=line[1] end if end if return result end function --------------------------------------------------------------------------- --returns the remainder of a string after word wrapping one line --this is used to do the word-wrapping of long-lines global function after_wrap_point(sequence string) sequence result integer size size=80 result=string[large(length(before_wrap_point(string)),1)+1..length(string)] if length(result) then if result[1]='\n' then result=decapitate(result) end if end if return result end function --------------------------------------------------------------------------- global function get_whole_file(sequence filename) integer fh sequence whole_file,chunk whole_file={} if file_exists(filename) then fh=open(filename,"rb") while true do chunk = get_bytes(fh,100) whole_file &= chunk if length(chunk) < 100 then exit end if end while close(fh) return(whole_file) else return({}) end if end function ---------------------------------------------------------------------------