#include #include #include #include using namespace std; using namespace Magick; Napi::Value Freeze(const Napi::CallbackInfo &info) { Napi::Env env = info.Env(); try { Napi::Object obj = info[0].As(); Napi::Buffer data = obj.Get("data").As>(); bool loop = obj.Has("loop") ? obj.Get("loop").As().Value() : false; string type = obj.Get("type").As().Utf8Value(); int delay = obj.Has("delay") ? obj.Get("delay").As().Int32Value() : 0; int frame = obj.Has("frame") ? obj.Get("frame").As().Int32Value() : -1; Napi::Object result = Napi::Object::New(env); char *fileData = data.Data(); char *match = (char *)"\x21\xFF\x0BNETSCAPE2.0\x03\x01"; char *descriptor = (char *)"\x2C\x00\x00\x00\x00"; char *lastPos; bool none = true; if (loop) { char *newData = (char *)malloc(data.Length() + 19); memcpy(newData, fileData, data.Length()); lastPos = (char *)memchr(newData, '\x2C', data.Length()); while (lastPos != NULL) { if (memcmp(lastPos, descriptor, 5) != 0) { lastPos = (char *)memchr(lastPos + 1, '\x2C', (data.Length() - (lastPos - newData)) - 1); continue; } memcpy(lastPos + 19, lastPos, (data.Length() - (lastPos - newData))); memcpy(lastPos, match, 16); memcpy(lastPos + 16, "\x00\x00\x00", 3); result.Set("data", Napi::Buffer::Copy(env, newData, data.Length() + 19)); none = false; break; } if (none) result.Set("data", Napi::Buffer::Copy(env, newData, data.Length())); } else if (frame >= 0 && !loop) { Blob blob; list frames; try { readImages(&frames, Blob(data.Data(), data.Length())); } catch (Magick::WarningCoder &warning) { cerr << "Coder Warning: " << warning.what() << endl; } catch (Magick::Warning &warning) { cerr << "Warning: " << warning.what() << endl; } size_t frameSize = frames.size(); int framePos = clamp(frame, 0, (int)frameSize); frames.resize(framePos + 1); for_each(frames.begin(), frames.end(), animationIterationsImage(loop ? 0 : 1)); for_each(frames.begin(), frames.end(), magickImage(type)); if (delay != 0) for_each(frames.begin(), frames.end(), animationDelayImage(delay)); writeImages(frames.begin(), frames.end(), &blob); result.Set("data", Napi::Buffer::Copy(env, (char *)blob.data(), blob.length())); } else { lastPos = (char *)memchr(fileData, '\x21', data.Length()); while (lastPos != NULL) { if (memcmp(lastPos, match, 16) != 0) { lastPos = (char *)memchr(lastPos + 1, '\x21', (data.Length() - (lastPos - fileData)) - 1); continue; } memcpy(lastPos, lastPos + 19, (data.Length() - (lastPos - fileData)) - 19); result.Set("data", Napi::Buffer::Copy(env, fileData, data.Length() - 19)); none = false; break; } if (none) result.Set("data", Napi::Buffer::Copy(env, fileData, data.Length())); } result.Set("type", type); return result; } catch (std::exception const &err) { throw Napi::Error::New(env, err.what()); } catch (...) { throw Napi::Error::New(env, "Unknown error"); } }