Example #1
0
        internal static WZObject ParseExtendedProperty(string name, WZBinaryReader r, WZObject parent, WZImage f,
                                                       bool encrypted)
        {
            string type = r.ReadWZStringBlock(encrypted);

            switch (type)
            {
            case "Property":
                r.Skip(2);
                return(new WZSubProperty(name, parent, r, f));

            case "Canvas":
                return(new WZCanvasProperty(name, parent, r, f));

            case "Shape2D#Vector2D":
                return(new WZPointProperty(name, parent, r, f));

            case "Shape2D#Convex2D":
                return(new WZConvexProperty(name, parent, r, f));

            case "Sound_DX8":
                return(new WZAudioProperty(name, parent, r, f));

            case "UOL":
                r.Skip(1);
                return(new WZLinkProperty(name, parent, r, f));

            default:
                return(WZUtil.Die <WZObject>($"Unknown ExtendedProperty type \"{type}\""));
            }
        }
Example #2
0
 private void Parse()
 {
     _r.Seek(0);
     if (_r.PeekFor(() => _r.ReadWZStringBlock(true)) == "Property")
     {
         _encrypted = true;
     }
     else if (_r.PeekFor(() => _r.ReadWZStringBlock(false)) == "Property")
     {
         _encrypted = false;
     }
     else
     {
         WZUtil.Die("Failed to determine image encryption!");
     }
     _r.SkipWZStringBlock();
     if (_r.ReadUInt16() != 0)
     {
         WZUtil.Die("WZImage with invalid header (no zero UInt16!)");
     }
     foreach (WZObject child in WZExtendedParser.ParsePropertyList(_r, this, this, _encrypted))
     {
         Add(child);
     }
     _parsed = true;
 }
        private static float ReadSingle(WZBinaryReader reader)
        {
            byte t = reader.ReadByte();

            return(t == 0x80
                ? reader.ReadSingle()
                : (t == 0 ? 0f : WZUtil.Die <float>("Unknown byte while reading WZSingleProperty.")));
        }
Example #4
0
        // write function
        /// <summary> 將指定位元組大小的資料寫入到緩衝區中,並將資料流位置往前移動 </summary>
        /// <param name="buffer"> 要寫入的資料 </param>
        /// <param name="length"> 要寫入的長度 </param>
        /// <param name="encrypt"> 是否要進行加密 </param>
        public void Write(byte[] buffer, int length, bool encrypt = false)
        {
            if (encrypt)
            {
                WZUtil.XORKey(buffer, length, this.KeyType);
            }

            this.BaseStream.Write(buffer, 0, length);
        }
Example #5
0
        // read functions
        /// <summary> 從目前的資料流讀取指定位元組大小的資料到指定的緩衝區內,並移動資料流指向的位置 </summary>
        /// <param name="buffer"> 接收資料的資料緩衝區 </param>
        /// <param name="length"> 需要讀取的資料長度 </param>
        /// <param name="decrypt"> 是否需要進行解密 </param>
        public void Read(byte[] buffer, int length, bool decrypt = false)
        {
            this.BaseStream.Read(buffer, 0, length);

            if (decrypt)
            {
                WZUtil.XORKey(buffer, length, this.KeyType);
            }
        }
Example #6
0
        internal static List <WZObject> ParsePropertyList(WZBinaryReader r, WZObject parent, WZImage f, bool encrypted)
        {
            int             num = r.ReadWZInt();
            List <WZObject> ret = new List <WZObject>(num);

            for (int i = 0; i < num; ++i)
            {
                string name = r.ReadWZStringBlock(encrypted);
                byte   type = r.ReadByte();
                switch (type)
                {
                case 0:
                    ret.Add(new WZNullProperty(name, parent, f));
                    break;

                case 0x0B:
                case 2:
                    ret.Add(new WZUInt16Property(name, parent, r, f));
                    break;

                case 0x13:
                case 3:
                    ret.Add(new WZInt32Property(name, parent, r, f));
                    break;

                case 0x14:
                    ret.Add(new WZInt64Property(name, parent, r, f));
                    break;

                case 4:
                    ret.Add(new WZSingleProperty(name, parent, r, f));
                    break;

                case 5:
                    ret.Add(new WZDoubleProperty(name, parent, r, f));
                    break;

                case 8:
                    ret.Add(new WZStringProperty(name, parent, r, f));
                    break;

                case 9:
                    uint blockLen = r.ReadUInt32();
                    ret.Add(ParseExtendedProperty(name, r.Clone(), parent, f, encrypted));
                    r.Skip(blockLen);
                    break;

                default:
                    return
                        (WZUtil.Die <List <WZObject> >(
                             $"Unknown property type {type} at ParsePropertyList"));
                }
            }
            return(ret);
        }
Example #7
0
        private void Parse(WZBinaryReader wzbr, long offset)
        {
            wzbr.Seek(offset);
            int entryCount = wzbr.ReadWZInt();

            for (int i = 0; i < entryCount; ++i)
            {
                byte   type = wzbr.ReadByte();
                string name = null;
                switch (type)
                {
                case 2:
                    int x = wzbr.ReadInt32();
                    wzbr.PeekFor(() => {
                        wzbr.Seek(x + File._fstart);
                        type = wzbr.ReadByte();
                        name = wzbr.ReadWZString(File._encrypted);
                    });

                    break;

                case 3:
                case 4:
                    name = wzbr.ReadWZString(File._encrypted);
                    break;

                case 1:
                // wzbr.Skip(10);
                // continue;
                default:
                    WZUtil.Die($"Unknown object type {type} in WzDirectory.");
                    break;
                }
                if (name == null)
                {
                    WZUtil.Die("Failed to read WZDirectory entry name.");
                }
                int size = wzbr.ReadWZInt();
                wzbr.ReadWZInt();
                uint woffset = wzbr.ReadWZOffset(File._fstart);
                switch (type)
                {
                case 3:
                    Add(new WZDirectory(name, this, File, wzbr.Clone(), woffset));
                    break;

                case 4:
                    Add(new WZImage(name, this, File,
                                    File.GetSubstream(woffset, size)));
                    break;
                }
            }
        }
Example #8
0
        /// <summary> </summary>
        public static uint GenerateOffsetKey(uint cur, uint off, int hash)
        {
            uint key = (uint)(((~(cur - off)) * hash) - 0x581C3F6D);

            return(WZUtil._rotl(key, (byte)(key & 0x1F)));
        }
        internal override bool Parse(WZBinaryReader br, bool initial, out Bitmap result)
        {
            bool skip  = (File._flag & WZReadSelection.NeverParseCanvas) == WZReadSelection.NeverParseCanvas,
                 eager = (File._flag & WZReadSelection.EagerParseCanvas) == WZReadSelection.EagerParseCanvas;

            if (initial)
            {
                if (skip && eager)
                {
                    result = null;
                    return(WZUtil.Die <bool>("Both NeverParseCanvas and EagerParseCanvas are set."));
                }
                br.Skip(1);
                if (br.ReadByte() == 1)
                {
                    br.Skip(2);
                    List <WZObject> l = WZExtendedParser.ParsePropertyList(br, this, Image, Image._encrypted);
                    if (ChildCount == 0)
                    {
                        l.ForEach(Add);
                    }
                }
                _afterChildren = br.Position;
            }
            else
            {
                br.Position = _afterChildren;
            }
            int width   = br.ReadWZInt(); // width
            int height  = br.ReadWZInt(); // height
            int format1 = br.ReadWZInt(); // format 1
            int scale   = br.ReadByte();  // format 2

            br.Skip(4);
            int blockLen = br.ReadInt32();

            if ((initial || skip) && !eager)
            {
                br.Skip(blockLen); // block Len & png data
                result = null;
                return(skip);
            }
            else
            {
                br.Skip(1);
                byte[] header = br.ReadBytes(2);
                // FLG bit 5 indicates the presence of a preset dictionary
                // seems like MS doesn't use that?
                if ((header[1] & 0x20) != 0)
                {
                    Debug.WriteLine("Warning; zlib with preset dictionary");
                    result = null;
                    br.Skip(blockLen - 3);
                    return(true);
                }
                // zlib header: CMF (byte), FLG (byte)
                byte[] pngData = br.ReadBytes(blockLen - 3);
                result = ParsePNG(width, height, format1, scale,
                                  // CMF least significant bits 0 to 3 are compression method. only 8 is valid
                                  (header[0] & 0xF) != 8 ||
                                  // CMF << 8 | FLG i.e. header read as a big-endian short is a multiple of 31
                                  (header[0] << 8 | header[1]) % 31 != 0
                        ? DecryptPNG(pngData)
                        : pngData);
                return(true);
            }
        }