/* Copyright (C) 2003-2015 LiveCode Ltd. This file is part of LiveCode. LiveCode is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License v3 as published by the Free Software Foundation. LiveCode is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with LiveCode. If not see . */ #include "prefix.h" #include "globdefs.h" #include "parsedef.h" #include "filedefs.h" #include "objdefs.h" #include "mcio.h" #include "object.h" #include "cdata.h" #include "field.h" #include "paragraf.h" MCCdata::MCCdata() : id(0), data(nil) { } MCCdata::MCCdata(const MCCdata &cref) : MCDLlist(cref) { // Ensure that the paragraphs of the cloned data have their parent field // set to nil - this will catch attempts to use them without properly // setting the parent first. CloneData(cref, nil); } MCCdata::MCCdata(const MCCdata& cref, MCField* p_new_owner) : MCDLlist(cref) { CloneData(cref, p_new_owner); } void MCCdata::CloneData(const MCCdata& cref, MCField* p_new_owner) { id = cref.id; if (cref.data != NULL && cref.data != (void *)1) { if (id & COMPACT_PARAGRAPHS) data = strclone((char *)cref.data); else { MCParagraph *paragraphs = NULL; MCParagraph *tptr = (MCParagraph *)cref.data; do { // Clone the paragraph MCParagraph *newparagraph = new (nothrow) MCParagraph(*tptr); newparagraph->setparent(p_new_owner); newparagraph->appendto(paragraphs); tptr = (MCParagraph *)tptr->next(); } while (tptr != cref.data); data = paragraphs; } } else data = cref.data; } MCCdata::MCCdata(uint4 newid) { id = newid; data = NULL; } MCCdata::~MCCdata() { if (data != NULL && data != (void *)1) { if (id & COMPACT_PARAGRAPHS) delete (char *)data; else { MCParagraph *paragraphs = (MCParagraph *)data; while (paragraphs != NULL) { MCParagraph *pptr = (MCParagraph *)paragraphs->remove (paragraphs); delete pptr; } } } } IO_stat MCCdata::load(IO_handle stream, MCObject *parent, uint32_t version) { IO_stat stat; if ((stat = IO_read_uint4(&id, stream)) != IO_NORMAL) return checkloadstat(stat); if (parent->gettype() == CT_BUTTON) { uint1 set; stat = IO_read_uint1(&set, stream); data = reinterpret_cast(set ? 1 : 0); return checkloadstat(stat); } else { if (id & COMPACT_PARAGRAPHS) { // MW-2013-11-19: [[ UnicodeFileFormat ]] This flag is never set by newer engines // so is just legacy. char *string; if ((stat = IO_read_cstring_legacy(string, stream, sizeof(uint1))) != IO_NORMAL) return checkloadstat(stat); data = string; } else { MCParagraph *paragraphs = NULL; while (True) { uint1 type; if ((stat = IO_read_uint1(&type, stream)) != IO_NORMAL) return checkloadstat(stat); switch (type) { // MW-2012-03-04: [[ StackFile5500 ]] Handle either the paragraph or extended // paragraph tag. case OT_PARAGRAPH: case OT_PARAGRAPH_EXT: { MCParagraph *newpar = new (nothrow) MCParagraph; newpar->setparent((MCField *)parent); // MW-2012-03-04: [[ StackFile5500 ]] If the paragraph tab was the extended // variant, then pass the correct is_ext parameter. if ((stat = newpar->load(stream, version, type == OT_PARAGRAPH_EXT)) != IO_NORMAL) { delete newpar; return checkloadstat(stat); } newpar->appendto(paragraphs); } break; default: data = paragraphs; MCS_seek_cur(stream, -1); return IO_NORMAL; } } } } return IO_NORMAL; } IO_stat MCCdata::save(IO_handle stream, Object_type type, uint4 p_part, MCObject *p_parent, uint32_t p_version) { IO_stat stat; // If p_part is non-zero it means we only want to save data specific // to a given card. In this case, we simply don't save the rest. if (p_part != 0 && id != p_part) return IO_NORMAL; if ((stat = IO_write_uint1(type, stream)) != IO_NORMAL) return stat; if ((stat = IO_write_uint4(p_part != 0 ? 0 : id, stream)) != IO_NORMAL) return stat; if (type == OT_BDATA) { uint1 set = data ? 1 : 0; return IO_write_uint1(set, stream); } else if (id & COMPACT_PARAGRAPHS) { // MW-2013-11-19: [[ UnicodeFileFormat ]] This flag is never set by newer engines // so is just legacy. (Indeed, this codepath should never be hit!). return IO_write_cstring_legacy((char *)data, stream, sizeof(uint1)); } else { MCParagraph *tptr = (MCParagraph *)data; if (tptr != NULL) do { // Ensure field's saved MCCdata paragraphs have a parent when needed if (p_parent != nil) tptr -> setparent(MCObjectCast(p_parent)); if ((stat = tptr->save(stream, p_part, p_version)) != IO_NORMAL) return stat; tptr = (MCParagraph *)tptr->next(); } while (tptr != data); } return IO_NORMAL; } uint4 MCCdata::getid() { return (id & ~COMPACT_PARAGRAPHS); } void MCCdata::setid(uint4 newid) { id = newid | (id & COMPACT_PARAGRAPHS); } MCParagraph *MCCdata::getparagraphs() { MCParagraph *paragraphs; if (id & COMPACT_PARAGRAPHS) { paragraphs = NULL; char *eptr = (char *)data; while ((eptr = strtok(eptr, "\n")) != NULL) { MCParagraph *tpgptr = new (nothrow) MCParagraph; tpgptr->appendto(paragraphs); uint2 l = strlen(eptr) + 1; /* UNCHECKED */ MCAutoPointer sptr = new (nothrow) char_t[l]; memcpy(sptr.Get(), eptr, l); MCAutoStringRef t_string; /* UNCHECKED */ MCStringCreateWithNativeChars(sptr.Get(), l, &t_string); tpgptr->settext(*t_string); eptr = NULL; } delete (char *)data; data = paragraphs; id &= ~COMPACT_PARAGRAPHS; } if (data == NULL) data = paragraphs = new (nothrow) MCParagraph; else paragraphs = (MCParagraph *)data; return paragraphs; } void MCCdata::setparagraphs(MCParagraph *&newpar) { id &= ~COMPACT_PARAGRAPHS; data = newpar; } Boolean MCCdata::getset() { return data != NULL; } void MCCdata::setset(Boolean newset) { data = (void *)(newset != 0); }