SheetProtection.cs 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. namespace Biff8Excel.Excel
  5. {
  6. public class SheetProtection : IDisposable
  7. {
  8. Records.Password m_password;
  9. Records.Protect m_protect;
  10. Records.ObjectProtection m_objProtect;
  11. Records.ScenProtect m_scenProtect;
  12. //Records.WindowsProtect m_winProtect;
  13. internal byte[] WriteRecord()
  14. {
  15. byte[] b = new byte[24]; // each record is 6 bytes long;
  16. // Protect Record;
  17. m_protect.GetByte().CopyTo(b, 0);
  18. // ScenProtect
  19. m_scenProtect.GetByte().CopyTo(b, 6);
  20. // objProtect
  21. m_objProtect.GetByte().CopyTo(b, 12);
  22. // Password
  23. m_password.GetByte().CopyTo(b, 18);
  24. return b;
  25. }
  26. internal void ProtectSheet(string password)
  27. {
  28. m_protect.SetProtect = 1;
  29. m_scenProtect.Protect = 1;
  30. m_objProtect.Protect = 1;
  31. if (password.Length > 0)
  32. m_password.PasswordHash = GetPassWordHash(password);
  33. }
  34. private ushort GetPassWordHash(string password)
  35. {
  36. ushort hash = 0;
  37. //ushort char_index = 0;
  38. ushort char_count = (ushort)password.Length;
  39. ushort chars;
  40. ushort y;
  41. for (int i = 0; i < char_count; i++)
  42. {
  43. chars = (ushort)(password.ToCharArray(i,1)[0]);
  44. y = RotateBit(chars, (ushort)(i+1));
  45. hash ^= y;
  46. }
  47. return (ushort)(hash ^ char_count ^ 0xCE4B);
  48. }
  49. private ushort RotateBit(ushort Value, ushort bit)
  50. {
  51. int temp = Value;
  52. ushort result;
  53. // Bit shift the value to the left
  54. temp <<= bit;
  55. // if any of the bits from 15 upwards are set
  56. // rotate them back to the start of the value
  57. for (int i = 15; i <= 30; i++)
  58. {
  59. //if ((temp & (1 << i)) == (1<<i))
  60. if ((temp & (1<<i)) != 0)
  61. {
  62. temp = (temp & (~(1 << i)));
  63. temp = (temp | (1 << (i - 15) ));
  64. }
  65. }
  66. // copy the first 2 bytes of the long value
  67. //result = BitConverter.ToInt16(BitConverter.GetBytes(temp), 0);
  68. result = BitConverter.ToUInt16(BitConverter.GetBytes(temp), 0);
  69. return result;
  70. }
  71. public SheetProtection()
  72. {
  73. m_password = new Biff8Excel.Records.Password();
  74. m_protect = new Biff8Excel.Records.Protect();
  75. m_objProtect = new Biff8Excel.Records.ObjectProtection();
  76. m_scenProtect = new Biff8Excel.Records.ScenProtect();
  77. }
  78. #region IDisposable ³ÉÔ±
  79. public void Dispose()
  80. {
  81. m_password = null;
  82. m_protect = null;
  83. m_objProtect = null;
  84. m_scenProtect = null;
  85. }
  86. #endregion
  87. }
  88. }