/*#io Sequence ioDoc( docCopyright("Steve Dekorte", 2002) docLicense("BSD revised") docObject("Sequence") docDescription("""A Sequence is a container for a list of data elements. Typically these elements are each 1 byte in size. A Sequence can be either mutable or immutable. When immutable, only the read-only methods can be used.

Terminology

""") docCategory("Core") */ #include "IoSeq.h" #include "IoState.h" #include "IoCFunction.h" #include "IoObject.h" #include "IoNumber.h" #include "IoMessage.h" #include "IoList.h" #include "IoMap.h" #include #include #define BIVAR(self) ((ByteArray *)IoObject_dataPointer(self)) #define IO_ASSERT_NOT_SYMBOL(self) \ if (ISSYMBOL(self)) \ { \ IoState_error_(IOSTATE, m, \ "'%s' cannot be called on an immutable Sequence", CSTRING(IoMessage_name(m)));\ } void IoSeq_addMutableMethods(IoSeq *self) { IoMethodTable methodTable[] = { {"copy", IoSeq_copy}, {"appendSeq", IoSeq_appendSeq}, {"append", IoSeq_append}, {"atInsertSeq", IoSeq_atInsertSeq}, {"removeSlice", IoSeq_removeSlice}, {"removeLast", IoSeq_removeLast}, {"setSize", IoSeq_setSize}, {"preallocateToSize", IoSeq_preallocateToSize}, {"replaceSeq", IoSeq_replaceSeq}, {"replaceFirstSeq", IoSeq_replaceFirstSeq}, {"atPut", IoSeq_atPut}, {"lowercase", IoSeq_lowercase}, {"uppercase", IoSeq_uppercase}, {"translate", IoSeq_translate}, {"clipBeforeSeq", IoSeq_clipBeforeSeq}, {"clipAfterSeq", IoSeq_clipAfterSeq}, {"clipBeforeEndOfSeq", IoSeq_clipBeforeEndOfSeq}, {"clipAfterStartOfSeq", IoSeq_clipAfterStartOfSeq}, {"empty", IoSeq_empty}, {"sort", IoSeq_sort}, {"reverse", IoSeq_reverse}, {"removeOddIndexes", IoSeq_removeOddIndexes}, {"removeEvenIndexes", IoSeq_removeEvenIndexes}, {"duplicateIndexes", IoSeq_duplicateIndexes}, {"replaceMap", IoSeq_replaceMap}, {"strip", IoSeq_strip}, {"lstrip", IoSeq_lstrip}, {"rstrip", IoSeq_rstrip}, {"float32ArrayAtPut", IoSeq_float32ArrayAtPut}, {"float32ArrayAdd", IoSeq_float32ArrayAdd}, {"float32ArrayMultiplyScalar", IoSeq_float32ArrayMultiplyScalar}, {"zero", IoSeq_zero}, {"escape", IoSeq_escape}, {"unescape", IoSeq_unescape}, {"removePrefix", IoSeq_removePrefix}, {"removeSuffix", IoSeq_removeSuffix}, {"capitalize", IoSeq_capitalize}, {"appendPathSeq", IoSeq_appendPathSeq}, {"interpolateInPlace", IoSeq_interpolateInPlace}, {NULL, NULL}, }; IoObject_addMethodTable_(self, methodTable); } IoObject *IoSeq_copy(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("copy(aSequence)", "Replaces the bytes of the receiver with a copy of those in aSequence. Returns self. ") */ IoObject *other; IO_ASSERT_NOT_SYMBOL(self); other = IoMessage_locals_seqArgAt_(m, locals, 0); ByteArray_copy_(BIVAR(self), BIVAR(other)); return self; } IoObject *IoSeq_appendSeq(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("appendSeq(aSequence1, aSequence2, ...)", "Appends aSequence arguments to the receiver. Returns self. ") */ int i; IO_ASSERT_NOT_SYMBOL(self); IOASSERT(IoMessage_argCount(m), "requires at least one argument"); for (i = 0; i < IoMessage_argCount(m); i ++) { IoObject *other = IoMessage_locals_valueArgAt_(m, locals, i); if (ISNUMBER(other)) { double d = IoNumber_asDouble(other); char s[24]; memset(s, 0, 24); if (d == (long)d) { snprintf(s, 24, "%ld", (long)d); } else { snprintf(s, 24, "%g", d); } ByteArray_appendCString_(BIVAR(self), s); } else if (ISSEQ(other)) { ByteArray_append_(BIVAR(self), BIVAR(other)); } else if (!ISNIL(other)) { IoState_error_(IOSTATE, m, "argument 0 to method '%s' must be a number, string or buffer, not a '%s'", CSTRING(IoMessage_name(m)), IoObject_name(other)); } } return self; } IoObject *IoSeq_append(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("append(aNumber)", "Appends aNumber (cast to a byte) to the receiver. Returns self. ") */ int i; IO_ASSERT_NOT_SYMBOL(self); IOASSERT(IoMessage_argCount(m), "requires at least one argument"); for (i = 0; i < IoMessage_argCount(m); i ++) { unsigned char s = (unsigned char)IoMessage_locals_intArgAt_(m, locals, i); ByteArray_appendByte_(BIVAR(self), s); } return self; } IoObject *IoSeq_atInsertSeq(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("atInsertSeq(indexNumber, aSequence)", "Returns a new Sequence with aSequence inserted at indexNumber in the receiver. ") */ int n = IoMessage_locals_intArgAt_(m, locals, 0); IoSeq *otherSeq = IoMessage_locals_seqArgAt_(m, locals, 1); IO_ASSERT_NOT_SYMBOL(self); ByteArray_insert_at_(BIVAR(self), BIVAR(otherSeq), n); return self; } // removing --------------------------------------- IoObject *IoSeq_removeSlice(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("removeSlice(startIndex, endIndex)", "Removes the bytes from startIndex to endIndex. Returns self.") */ int start = IoMessage_locals_intArgAt_(m, locals, 0); int end = IoMessage_locals_intArgAt_(m, locals, 1); IO_ASSERT_NOT_SYMBOL(self); ByteArray_removeSlice(BIVAR(self), start, end + 1); return self; } IoObject *IoSeq_removeLast(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("removeLast", "Removes the last element from the receiver. Returns self.") */ IO_ASSERT_NOT_SYMBOL(self); ByteArray_dropLastByte(BIVAR(self)); return self; } IoObject *IoSeq_setSize(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("setSize(aNumber)", "Sets the length in bytes of the receiver to aNumber. Return self.") */ int len = IoMessage_locals_intArgAt_(m, locals, 0); IO_ASSERT_NOT_SYMBOL(self); IOASSERT(len >= 0, "New size cannot be negative."); ByteArray_setSize_(BIVAR(self), len); return self; } void IoSeq_rawPreallocateToSize_(IoSeq *self, size_t size) { if (ISSYMBOL(self)) { IoState_error_(IOSTATE, NULL, "attempt to resize an immutable Sequence"); } ByteArray_sizeTo_(BIVAR(self), size); } IoObject *IoSeq_preallocateToSize(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("preallocateToSize(aNumber)", "If needed, resize the memory alloced for the receivers byte array to be large enough to fit the number of bytes specified by aNumber. This is useful for preallocating the memory so it doesn't keep getting allocated as the Sequence is appended to. This operation will not change the Sequence's length or contents. Returns self.") */ int newSize = IoMessage_locals_intArgAt_(m, locals, 0); IO_ASSERT_NOT_SYMBOL(self); IOASSERT(newSize >= 0, "New size cannot be negative."); ByteArray_sizeTo_(BIVAR(self), newSize); return self; } IoObject *IoSeq_replaceSeq(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("replaceSeq(aSequence, anotherSequence)", "Returns a new Sequence with all occurances of aSequence replaced with anotherSequence in the receiver. Returns self.") */ IoSeq *subSeq = IoMessage_locals_seqArgAt_(m, locals, 0); IoSeq *otherSeq = IoMessage_locals_seqArgAt_(m, locals, 1); IO_ASSERT_NOT_SYMBOL(self); ByteArray_replace_with_(BIVAR(self), BIVAR(subSeq), BIVAR(otherSeq)); return self; } IoObject *IoSeq_replaceFirstSeq(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("replaceFirstSeq(aSequence, anotherSequence, optionalStartIndex)", "Returns a new Sequence with the first occurance of aSequence replaced with anotherSequence in the receiver. If optionalStartIndex is provided, the search for aSequence begins at that index. Returns self.") */ IoSeq *subSeq = IoMessage_locals_seqArgAt_(m, locals, 0); IoSeq *otherSeq = IoMessage_locals_seqArgAt_(m, locals, 1); size_t startIndex = 0; if (IoMessage_argCount(m) > 2) { IoMessage_locals_longArgAt_(m, locals, 1); } IO_ASSERT_NOT_SYMBOL(self); ByteArray_replaceFirst_from_with_(BIVAR(self), BIVAR(subSeq), startIndex, BIVAR(otherSeq)); return self; } IoObject *IoSeq_atPut(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("atPut(aNumberIndex, aNumberByte)", "Sets the byte at the index specified by aNumberIndex to the byte value of aNumberByte. Returns self. ") */ int i = IoMessage_locals_intArgAt_(m, locals, 0); int v = IoMessage_locals_intArgAt_(m, locals, 1); IO_ASSERT_NOT_SYMBOL(self); ByteArray_at_put_(BIVAR(self), (int)i, (unsigned char)v); return self; } IoObject *IoSeq_lowercase(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("Lowercase", "Returns a copy of the receiver with all characters made Lowercase. ") */ IO_ASSERT_NOT_SYMBOL(self); ByteArray_Lowercase(BIVAR(self)); return self; } IoObject *IoSeq_uppercase(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("uppercase", "Makes all characters of the receiver uppercase. ") */ IO_ASSERT_NOT_SYMBOL(self); ByteArray_uppercase(BIVAR(self)); return self; } // clip -------------------------------------- IoObject *IoSeq_clipBeforeSeq(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("clipBeforeSeq(aSequence)", "Clips receiver before aSequence.") */ IoSeq *otherSeq = IoMessage_locals_seqArgAt_(m, locals, 0); IO_ASSERT_NOT_SYMBOL(self); ByteArray_clipBefore_(BIVAR(self), BIVAR(otherSeq)); return self; } IoObject *IoSeq_clipAfterSeq(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("clipAfterSeq(aSequence)", "Removes the contents of the receiver after the end of the first occurance of aSequence. Returns true if anything was removed, or false otherwise.") */ IoSeq *otherSeq = IoMessage_locals_seqArgAt_(m, locals, 0); IO_ASSERT_NOT_SYMBOL(self); ByteArray_clipAfter_(BIVAR(self), BIVAR(otherSeq)); return self; } IoObject *IoSeq_clipBeforeEndOfSeq(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("clipBeforeEndOfSeq(aSequence)", "Removes the contents of the receiver before the end of the first occurance of aSequence. Returns true if anything was removed, or false otherwise.") */ IoSeq *otherSeq = IoMessage_locals_seqArgAt_(m, locals, 0); IO_ASSERT_NOT_SYMBOL(self); ByteArray_clipBeforeEndOf_(BIVAR(self), BIVAR(otherSeq)); return self; } IoObject *IoSeq_clipAfterStartOfSeq(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("clipAfterStartOfSeq(aSequence)", "Removes the contents of the receiver after the beginning of the first occurance of aSequence. Returns true if anything was removed, or false otherwise.") */ IoSeq *otherSeq = IoMessage_locals_seqArgAt_(m, locals, 0); IO_ASSERT_NOT_SYMBOL(self); ByteArray_clipAfterStartOf_(BIVAR(self), BIVAR(otherSeq)); return self; } // ----------------------------------------- IoObject *IoSeq_empty(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("clear", "Sets all bytes in the receiver to 0x0 and sets it's length to 0. Returns self.") */ IO_ASSERT_NOT_SYMBOL(self); ByteArray_clear(BIVAR(self)); ByteArray_setSize_(BIVAR(self), 0); return self; } int IoSeq_byteCompare(const void *a, const void *b) { char aa = *(char *)a; char bb = *(char *)b; if (aa < bb) { return -1; } if (aa == bb) { return 0; } return 1; } IoObject *IoSeq_sort(IoSeq *self, IoObject *locals, IoMessage *m) { IO_ASSERT_NOT_SYMBOL(self); qsort(ByteArray_bytes(BIVAR(self)), ByteArray_size(BIVAR(self)), 1, IoSeq_byteCompare); return self; } IoObject *IoSeq_removeOddIndexes(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("removeOddIndexes", "Removes odd indexes in the receiver. For example, list(1,2,3) removeOddIndexes == list(2). Returns self.") */ int size = IoMessage_locals_intArgAt_(m, locals, 0); IO_ASSERT_NOT_SYMBOL(self); ByteArray_removeOddIndexesOfSize_(BIVAR(self), size); return self; } IoObject *IoSeq_removeEvenIndexes(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("removeEvenIndexes", "Removes even indexes in the receiver. For example, list(1,2,3) removeEvenIndexes == list(1, 3). Returns self.") */ int size = IoMessage_locals_intArgAt_(m, locals, 0); IO_ASSERT_NOT_SYMBOL(self); ByteArray_removeEvenIndexesOfSize_(BIVAR(self), size); return self; } IoObject *IoSeq_duplicateIndexes(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("duplicateIndexes", "Duplicates all indexes in the receiver. For example, list(1,2,3) duplicateIndexes == list(1,1,2,2,3,3). Returns self.") */ int size = IoMessage_locals_intArgAt_(m, locals, 0); IO_ASSERT_NOT_SYMBOL(self); ByteArray_duplicateIndexesOfSize_(BIVAR(self), size); return self; } IoObject *IoSeq_replaceMap(IoObject *self, IoObject *locals, IoMessage *m) { /*#io docSlot("replaceMap(aMap)", "In the receiver, the keys of aMap replaced with it's values. Returns self.") */ IoMap *map = IoMessage_locals_mapArgAt_(m, locals, 0); Hash *hash = IoMap_rawHash(map); IoSymbol *subSeq = Hash_firstKey(hash); ByteArray *ba = BIVAR(self); IO_ASSERT_NOT_SYMBOL(self); while (subSeq) { IoSymbol *otherSeq = Hash_at_(hash, subSeq); if (ISSEQ(otherSeq)) { ByteArray_replace_with_(ba, BIVAR(subSeq), BIVAR(otherSeq)); } else { IoState_error_(IOSTATE, m, "argument 0 to method '%s' must be a Map with Sequence values, not '%s'", CSTRING(IoMessage_name(m)), IoObject_name(otherSeq)); } subSeq = Hash_nextKey(hash); } return self; } // translate ------------------------------------------------------ IoObject *IoSeq_translate(IoObject *self, IoObject *locals, IoMessage *m) { /*#io docSlot("translate(fromChars, toChars)", "In the receiver, the characters in fromChars are replaced with those in the same positions in toChars. Returns self.") */ ByteArray *tc = BIVAR(IoMessage_locals_seqArgAt_(m, locals, 1)); ByteArray *fc = BIVAR(IoMessage_locals_seqArgAt_(m, locals, 0)); int tc_len = ByteArray_size(tc); int fc_len = ByteArray_size(fc); ByteArray *ba = BIVAR(self); char lut[256], currChar; int i; IO_ASSERT_NOT_SYMBOL(self); if (fc_len != tc_len) IoState_error_(IOSTATE, m, "translation strings must be of the same length"); // Build the lookup table: // for each char in fc, set lut[fc[i]] = bc[i] for (i = 0; i <= 255; i ++) { lut[i] = i; } // init the lookup table for (i = 0; i < fc_len; i ++) { lut[(char)ByteArray_at_(fc, i)] = ByteArray_at_(tc,i); //printf("%i\n", lut[i]); } // Step through the receiver's bytearray and make substitions, // replacing each char with the replacement from the lut for (i = 0; i < ByteArray_size(ba); i ++) { currChar = ByteArray_at_(ba, i); //printf("%c->%c\n", currChar, lut[currChar]); ba->bytes[i] = lut[currChar]; // replace } return self; } // reverse -------------------------------------------------------- IoObject *IoSeq_reverse(IoObject *self, IoObject *locals, IoMessage *m) { /*#io docSlot("reverse", "Reverses the bytes in the receiver, in-place.") */ int c, i, j; ByteArray *ba = BIVAR(self); IO_ASSERT_NOT_SYMBOL(self); for (i=0, j=ByteArray_size(ba)-1; i < j; i++, j--) { c = ba->bytes[i]; ba->bytes[i] = ba->bytes[j]; ba->bytes[j] = c; } return self; } // strip ---------------------------------------------------------- IoObject *IoSeq_strip(IoObject *self, IoObject *locals, IoMessage *m) { /*#io docSlot("strip(optionalSequence)", """Trims the whitespace (or optionalSequence) off both ends:
"   Trim this string   \r\n" strip
==> "Trim this string"
""") */ IO_ASSERT_NOT_SYMBOL(self); if (IoMessage_argCount(m) > 0) { IoSeq *other = IoMessage_locals_seqArgAt_(m, locals, 0); ByteArray_strip_(BIVAR(self), BIVAR(other)); } else { ByteArray *space = ByteArray_newWithCString_(WHITESPACE); ByteArray_strip_(BIVAR(self), space); ByteArray_free(space); } return self; } IoObject *IoSeq_lstrip(IoObject *self, IoObject *locals, IoMessage *m) { /*#io docSlot("lstrip(aSequence)", """Strips the characters in aSequence stripped from the beginning of the receiver. Example:
"Keep the tail" lstrip(" eKp")
==> "the tail"
""") */ IO_ASSERT_NOT_SYMBOL(self); if (IoMessage_argCount(m) > 0) { IoSeq *other = IoMessage_locals_seqArgAt_(m, locals, 0); ByteArray_lstrip_(BIVAR(self), BIVAR(other)); } else { ByteArray *space = ByteArray_newWithCString_(WHITESPACE); ByteArray_lstrip_(BIVAR(self), space); ByteArray_free(space); } return self; } IoObject *IoSeq_rstrip(IoObject *self, IoObject *locals, IoMessage *m) { /*#io docSlot("rstrip(aSequence)", """Strips the characters in aSequence stripped from the end of the receiver. Example:
"Cut the tail off" rstrip(" afilot")
==> "Cut the"
""") */ IO_ASSERT_NOT_SYMBOL(self); if (IoMessage_argCount(m) > 0) { IoSeq *other = IoMessage_locals_seqArgAt_(m, locals, 0); ByteArray_rstrip_(BIVAR(self), BIVAR(other)); } else { ByteArray *space = ByteArray_newWithCString_(WHITESPACE); ByteArray_rstrip_(BIVAR(self), space); ByteArray_free(space); } return self; } // float IoObject *IoSeq_float32ArrayAtPut(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("float32ArrayAtPut(value, indexNumber)", "Treats the buffer as an array of 32 bit floats and returns the Number at the specified index.") */ float v = IoMessage_locals_floatArgAt_(m, locals, 0); int index = IoMessage_locals_intArgAt_(m, locals, 1); int length = ByteArray_size32(BIVAR(self)); IOASSERT(index < length, "index out of bounds"); IO_ASSERT_NOT_SYMBOL(self); ByteArray_setFloat32_at_(BIVAR(self), v, index); return self; } IoObject *IoSeq_float32ArrayAdd(IoSeq *self, IoObject *locals, IoMessage *m) { IoSeq *other = IoMessage_locals_seqArgAt_(m, locals, 0); ByteArray_float32ArrayAdd_(IoSeq_rawByteArray(self), IoSeq_rawByteArray(other)); return self; } IoObject *IoSeq_float32ArrayMultiplyScalar(IoSeq *self, IoObject *locals, IoMessage *m) { float s = (float)IoMessage_locals_doubleArgAt_(m, locals, 0); ByteArray_float32ArrayMultiplyScalar_(IoSeq_rawByteArray(self), s); return self; } IoObject *IoSeq_zero(IoSeq *self, IoObject *locals, IoMessage *m) { ByteArray_zero(IoSeq_rawByteArray(self)); return self; } // ----------------------------------------------------------- IoObject *IoSeq_escape(IoObject *self, IoObject *locals, IoMessage *m) { /*#io docSlot("escape", """Escape characters in the receiver are replaced with escape codes. For example a string containing a single return character would contain the following 2 characters after being escaped: "\n". Returns self.""") */ IO_ASSERT_NOT_SYMBOL(self); ByteArray_escape(BIVAR(self)); return self; } IoObject *IoSeq_unescape(IoObject *self, IoObject *locals, IoMessage *m) { /*#io docSlot("unescape", "Escape codes replaced with escape characters. Returns self.") */ IO_ASSERT_NOT_SYMBOL(self); ByteArray_unescape(BIVAR(self)); return self; } IoObject *IoSeq_removePrefix(IoObject *self, IoObject *locals, IoMessage *m) { /*#io docSlot("removePrefix(aSequence)", "If the receiver begins with aSequence, it is removed. Returns self.") */ IoSeq *other = IoMessage_locals_seqArgAt_(m, locals, 0); IO_ASSERT_NOT_SYMBOL(self); if (ByteArray_beginsWith_(BIVAR(self), BIVAR(other))) { ByteArray_removeSlice(BIVAR(self), 0, ByteArray_size(BIVAR(other))); } return self; } IoObject *IoSeq_removeSuffix(IoObject *self, IoObject *locals, IoMessage *m) { /*#io docSlot("removeSuffix(aSequence)", "If the receiver end with aSequence, it is removed. Returns self.") */ IoSeq *other = IoMessage_locals_seqArgAt_(m, locals, 0); IO_ASSERT_NOT_SYMBOL(self); if (ByteArray_endsWith_(BIVAR(self), BIVAR(other))) { ByteArray *ba = BIVAR(self); ByteArray_removeSlice(ba, ByteArray_size(ba) - ByteArray_size(BIVAR(other)), ByteArray_size(ba)); } return self; } IoObject *IoSeq_capitalize(IoObject *self, IoObject *locals, IoMessage *m) { /*#io docSlot("capitalize", "First charater of the receiver is made uppercase.") */ char firstChar = ByteArray_at_(BIVAR(self), 0); IO_ASSERT_NOT_SYMBOL(self); ByteArray_at_put_(BIVAR(self), 0, toupper(firstChar)); return self; } IoObject *IoSeq_appendPathSeq(IoObject *self, IoObject *locals, IoMessage *m) { /*#io docSlot("appendPathSeq(aSeq)", "Appends argument to the receiver such that there is one and only one path separator between the two. Returns self.") */ IoSeq *component = IoMessage_locals_seqArgAt_(m, locals, 0); IO_ASSERT_NOT_SYMBOL(self); ByteArray_appendPathCString_(BIVAR(self), (char *)BIVAR(component)->bytes); return self; } IoObject *IoSeq_interpolateInPlace(IoSeq *self, IoObject *locals, IoMessage *m) { /*#io docSlot("interpolateInPlace(ctx)", "Replaces all #{} with evaluated in the context ctx. " "If ctx not given the current context is used. Returns self.") */ IoObject *context; ByteArray *string; ByteArray *code; IoObject *evaluatedCode; ByteArray *evaluatedCodeAsString; const char *label; int from, to; IO_ASSERT_NOT_SYMBOL(self); context = IoMessage_locals_valueArgAt_(m, locals, 0); string = BIVAR(self); label = "IoSeq_interpolateInPlace()"; from = 0; context = (context == IONIL(self)) ? locals : context; for(;;) { from = ByteArray_findCString_from_(string, "#{", from); if (from == -1) break; to = ByteArray_findCString_from_(string, "}", from); if (to == -1) break; code = ByteArray_newWithBytesFrom_to_(string, from+2, to); if (ByteArray_size(code) == 0) { // we don't want "#{}" to interpolate into "nil" evaluatedCodeAsString = BIVAR(IoState_doCString_(IOSTATE, "Sequence clone")); } else { evaluatedCode = IoState_on_doCString_withLabel_(IOSTATE, context, (char *)ByteArray_bytes(code), label); evaluatedCodeAsString = BIVAR(IoState_on_doCString_withLabel_(IOSTATE, evaluatedCode, "asString", label)); } ByteArray_free(code); code = NULL; ByteArray_replaceFrom_size_with_(string, from, to-from+1, evaluatedCodeAsString); } return self; }