Skip to content

Commit 1b7230e

Browse files
committedAug 12, 2022
fix(equipmentCloud): add rgif.js gif.js
1 parent 350b986 commit 1b7230e

File tree

6 files changed

+515
-11
lines changed

6 files changed

+515
-11
lines changed
 

‎CHANGELOG.EN.md

+1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@
77
* **appveyor:** script ([ce59370](https://github.com/dgiot/dgiot-dashboard/commit/ce593705fafa3c5b16062d27f43d66034139d561))
88
* **cloudTest:** evidence ([e8d0510](https://github.com/dgiot/dgiot-dashboard/commit/e8d051073aee73634a11fe3d31f0a7c4f54f67be))
99
* **dashboard): build(发布版本:** dgiot-dashboard@v4.5.6 ([1d4302c](https://github.com/dgiot/dgiot-dashboard/commit/1d4302cd183474db6bbbb6dbc4fbc46163d351c8)), closes [#511](https://github.com/dgiot/dgiot-dashboard/issues/511)
10+
* **equipmentCloud:** 通道编辑上传文件 ([53d1e6d](https://github.com/dgiot/dgiot-dashboard/commit/53d1e6d269ee2441bc2d13384f429f42dd80d1f6))
1011
* **equipmentCloud:** 字典修复 ([24a9690](https://github.com/dgiot/dgiot-dashboard/commit/24a96904b4879da9a952c9f6bf7358048863198c))
1112
* **ignore:** hidden test_script ([61b6326](https://github.com/dgiot/dgiot-dashboard/commit/61b632636fc8409b1e99f4235a9a865d22d72af4))
1213
* **queryZetaProduct:** queryProductTemplet ([1c47a45](https://github.com/dgiot/dgiot-dashboard/commit/1c47a4511451818ecd1f8fdcf2b4cebd0e6b4761))

‎CHANGELOG.ZH.md

+1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@
77
* **appveyor:** script ([ce59370](https://github.com/dgiot/dgiot-dashboard/commit/ce593705fafa3c5b16062d27f43d66034139d561))
88
* **cloudTest:** evidence ([e8d0510](https://github.com/dgiot/dgiot-dashboard/commit/e8d051073aee73634a11fe3d31f0a7c4f54f67be))
99
* **dashboard): build(发布版本:** dgiot-dashboard@v4.5.6 ([1d4302c](https://github.com/dgiot/dgiot-dashboard/commit/1d4302cd183474db6bbbb6dbc4fbc46163d351c8)), closes [#511](https://github.com/dgiot/dgiot-dashboard/issues/511)
10+
* **equipmentCloud:** 通道编辑上传文件 ([53d1e6d](https://github.com/dgiot/dgiot-dashboard/commit/53d1e6d269ee2441bc2d13384f429f42dd80d1f6))
1011
* **equipmentCloud:** 字典修复 ([24a9690](https://github.com/dgiot/dgiot-dashboard/commit/24a96904b4879da9a952c9f6bf7358048863198c))
1112
* **ignore:** hidden test_script ([61b6326](https://github.com/dgiot/dgiot-dashboard/commit/61b632636fc8409b1e99f4235a9a865d22d72af4))
1213
* **queryZetaProduct:** queryProductTemplet ([1c47a45](https://github.com/dgiot/dgiot-dashboard/commit/1c47a4511451818ecd1f8fdcf2b4cebd0e6b4761))

‎CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@
77
* **appveyor:** script ([ce59370](https://github.com/dgiot/dgiot-dashboard/commit/ce593705fafa3c5b16062d27f43d66034139d561))
88
* **cloudTest:** evidence ([e8d0510](https://github.com/dgiot/dgiot-dashboard/commit/e8d051073aee73634a11fe3d31f0a7c4f54f67be))
99
* **dashboard): build(发布版本:** dgiot-dashboard@v4.5.6 ([1d4302c](https://github.com/dgiot/dgiot-dashboard/commit/1d4302cd183474db6bbbb6dbc4fbc46163d351c8)), closes [#511](https://github.com/dgiot/dgiot-dashboard/issues/511)
10+
* **equipmentCloud:** 通道编辑上传文件 ([53d1e6d](https://github.com/dgiot/dgiot-dashboard/commit/53d1e6d269ee2441bc2d13384f429f42dd80d1f6))
1011
* **equipmentCloud:** 字典修复 ([24a9690](https://github.com/dgiot/dgiot-dashboard/commit/24a96904b4879da9a952c9f6bf7358048863198c))
1112
* **ignore:** hidden test_script ([61b6326](https://github.com/dgiot/dgiot-dashboard/commit/61b632636fc8409b1e99f4235a9a865d22d72af4))
1213
* **queryZetaProduct:** queryProductTemplet ([1c47a45](https://github.com/dgiot/dgiot-dashboard/commit/1c47a4511451818ecd1f8fdcf2b4cebd0e6b4761))

‎src/utils/file/gif.js

+364
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,364 @@
1+
// Generic functions
2+
/* eslint-disable */
3+
var bitsToNum = function (ba) {
4+
return ba.reduce(function (s, n) {
5+
return s * 2 + n
6+
}, 0)
7+
}
8+
9+
var byteToBitArr = function (bite) {
10+
var a = []
11+
for (var i = 7; i >= 0; i--) {
12+
a.push(!!(bite & (1 << i)))
13+
}
14+
return a
15+
}
16+
17+
// Stream
18+
/**
19+
* @constructor
20+
*/ // Make compiler happy.
21+
var Stream = function (data) {
22+
this.data = data
23+
this.len = this.data.length
24+
this.pos = 0
25+
26+
this.readByte = function () {
27+
if (this.pos >= this.data.length) {
28+
throw new Error('Attempted to read past end of stream.')
29+
}
30+
return data.charCodeAt(this.pos++) & 0xff
31+
}
32+
33+
this.readBytes = function (n) {
34+
var bytes = []
35+
for (var i = 0; i < n; i++) {
36+
bytes.push(this.readByte())
37+
}
38+
return bytes
39+
}
40+
41+
this.read = function (n) {
42+
var s = ''
43+
for (var i = 0; i < n; i++) {
44+
s += String.fromCharCode(this.readByte())
45+
}
46+
return s
47+
}
48+
49+
this.readUnsigned = function () {
50+
// Little-endian.
51+
var a = this.readBytes(2)
52+
return (a[1] << 8) + a[0]
53+
}
54+
}
55+
56+
var lzwDecode = function (minCodeSize, data) {
57+
// TODO: Now that the GIF parser is a bit different, maybe this should get an array of bytes instead of a String?
58+
var pos = 0 // Maybe this streaming thing should be merged with the Stream?
59+
60+
var readCode = function (size) {
61+
var code = 0
62+
for (var i = 0; i < size; i++) {
63+
if (data.charCodeAt(pos >> 3) & (1 << (pos & 7))) {
64+
code |= 1 << i
65+
}
66+
pos++
67+
}
68+
return code
69+
}
70+
71+
var output = []
72+
73+
var clearCode = 1 << minCodeSize
74+
var eoiCode = clearCode + 1
75+
76+
var codeSize = minCodeSize + 1
77+
78+
var dict = []
79+
80+
var clear = function () {
81+
dict = []
82+
codeSize = minCodeSize + 1
83+
for (var i = 0; i < clearCode; i++) {
84+
dict[i] = [i]
85+
}
86+
dict[clearCode] = []
87+
dict[eoiCode] = null
88+
}
89+
90+
var code
91+
var last
92+
93+
while (true) {
94+
last = code
95+
code = readCode(codeSize)
96+
97+
if (code === clearCode) {
98+
clear()
99+
continue
100+
}
101+
if (code === eoiCode) break
102+
103+
if (code < dict.length) {
104+
if (last !== clearCode) {
105+
dict.push(dict[last].concat(dict[code][0]))
106+
}
107+
} else {
108+
if (code !== dict.length) throw new Error('Invalid LZW code.')
109+
dict.push(dict[last].concat(dict[last][0]))
110+
}
111+
output.push.apply(output, dict[code])
112+
113+
if (dict.length === 1 << codeSize && codeSize < 12) {
114+
// If we're at the last code and codeSize is 12, the next code will be a clearCode, and it'll be 12 bits long.
115+
codeSize++
116+
}
117+
}
118+
119+
// I don't know if this is technically an error, but some GIFs do it.
120+
//if (Math.ceil(pos / 8) !== data.length) throw new Error('Extraneous LZW bytes.');
121+
return output
122+
}
123+
124+
// The actual parsing; returns an object with properties.
125+
var parseGIF = function (st, handler) {
126+
handler || (handler = {})
127+
128+
// LZW (GIF-specific)
129+
var parseCT = function (entries) {
130+
// Each entry is 3 bytes, for RGB.
131+
var ct = []
132+
for (var i = 0; i < entries; i++) {
133+
ct.push(st.readBytes(3))
134+
}
135+
return ct
136+
}
137+
138+
var readSubBlocks = function () {
139+
var size, data
140+
data = ''
141+
do {
142+
size = st.readByte()
143+
data += st.read(size)
144+
} while (size !== 0)
145+
return data
146+
}
147+
148+
var parseHeader = function () {
149+
var hdr = {}
150+
hdr.sig = st.read(3)
151+
hdr.ver = st.read(3)
152+
if (hdr.sig !== 'GIF') throw new Error('Not a GIF file.') // XXX: This should probably be handled more nicely.
153+
154+
hdr.width = st.readUnsigned()
155+
hdr.height = st.readUnsigned()
156+
157+
var bits = byteToBitArr(st.readByte())
158+
hdr.gctFlag = bits.shift()
159+
hdr.colorRes = bitsToNum(bits.splice(0, 3))
160+
hdr.sorted = bits.shift()
161+
hdr.gctSize = bitsToNum(bits.splice(0, 3))
162+
163+
hdr.bgColor = st.readByte()
164+
hdr.pixelAspectRatio = st.readByte() // if not 0, aspectRatio = (pixelAspectRatio + 15) / 64
165+
166+
if (hdr.gctFlag) {
167+
hdr.gct = parseCT(1 << (hdr.gctSize + 1))
168+
}
169+
handler.hdr && handler.hdr(hdr)
170+
}
171+
172+
var parseExt = function (block) {
173+
var parseGCExt = function (block) {
174+
var blockSize = st.readByte() // Always 4
175+
176+
var bits = byteToBitArr(st.readByte())
177+
block.reserved = bits.splice(0, 3) // Reserved; should be 000.
178+
block.disposalMethod = bitsToNum(bits.splice(0, 3))
179+
block.userInput = bits.shift()
180+
block.transparencyGiven = bits.shift()
181+
182+
block.delayTime = st.readUnsigned()
183+
184+
block.transparencyIndex = st.readByte()
185+
186+
block.terminator = st.readByte()
187+
188+
handler.gce && handler.gce(block)
189+
}
190+
191+
var parseComExt = function (block) {
192+
block.comment = readSubBlocks()
193+
handler.com && handler.com(block)
194+
}
195+
196+
var parsePTExt = function (block) {
197+
// No one *ever* uses this. If you use it, deal with parsing it yourself.
198+
var blockSize = st.readByte() // Always 12
199+
block.ptHeader = st.readBytes(12)
200+
block.ptData = readSubBlocks()
201+
handler.pte && handler.pte(block)
202+
}
203+
204+
var parseAppExt = function (block) {
205+
var parseNetscapeExt = function (block) {
206+
var blockSize = st.readByte() // Always 3
207+
block.unknown = st.readByte() // ??? Always 1? What is this?
208+
block.iterations = st.readUnsigned()
209+
block.terminator = st.readByte()
210+
handler.app && handler.app.NETSCAPE && handler.app.NETSCAPE(block)
211+
}
212+
213+
var parseUnknownAppExt = function (block) {
214+
block.appData = readSubBlocks()
215+
// FIXME: This won't work if a handler wants to match on any identifier.
216+
handler.app &&
217+
handler.app[block.identifier] &&
218+
handler.app[block.identifier](block)
219+
}
220+
221+
var blockSize = st.readByte() // Always 11
222+
block.identifier = st.read(8)
223+
block.authCode = st.read(3)
224+
switch (block.identifier) {
225+
case 'NETSCAPE':
226+
parseNetscapeExt(block)
227+
break
228+
default:
229+
parseUnknownAppExt(block)
230+
break
231+
}
232+
}
233+
234+
var parseUnknownExt = function (block) {
235+
block.data = readSubBlocks()
236+
handler.unknown && handler.unknown(block)
237+
}
238+
239+
block.label = st.readByte()
240+
switch (block.label) {
241+
case 0xf9:
242+
block.extType = 'gce'
243+
parseGCExt(block)
244+
break
245+
case 0xfe:
246+
block.extType = 'com'
247+
parseComExt(block)
248+
break
249+
case 0x01:
250+
block.extType = 'pte'
251+
parsePTExt(block)
252+
break
253+
case 0xff:
254+
block.extType = 'app'
255+
parseAppExt(block)
256+
break
257+
default:
258+
block.extType = 'unknown'
259+
parseUnknownExt(block)
260+
break
261+
}
262+
}
263+
264+
var parseImg = function (img) {
265+
var deinterlace = function (pixels, width) {
266+
// Of course this defeats the purpose of interlacing. And it's *probably*
267+
// the least efficient way it's ever been implemented. But nevertheless...
268+
269+
var newPixels = new Array(pixels.length)
270+
var rows = pixels.length / width
271+
var cpRow = function (toRow, fromRow) {
272+
var fromPixels = pixels.slice(fromRow * width, (fromRow + 1) * width)
273+
newPixels.splice.apply(
274+
newPixels,
275+
[toRow * width, width].concat(fromPixels)
276+
)
277+
}
278+
279+
// See appendix E.
280+
var offsets = [0, 4, 2, 1]
281+
var steps = [8, 8, 4, 2]
282+
283+
var fromRow = 0
284+
for (var pass = 0; pass < 4; pass++) {
285+
for (var toRow = offsets[pass]; toRow < rows; toRow += steps[pass]) {
286+
cpRow(toRow, fromRow)
287+
fromRow++
288+
}
289+
}
290+
291+
return newPixels
292+
}
293+
294+
img.leftPos = st.readUnsigned()
295+
img.topPos = st.readUnsigned()
296+
img.width = st.readUnsigned()
297+
img.height = st.readUnsigned()
298+
299+
var bits = byteToBitArr(st.readByte())
300+
img.lctFlag = bits.shift()
301+
img.interlaced = bits.shift()
302+
img.sorted = bits.shift()
303+
img.reserved = bits.splice(0, 2)
304+
img.lctSize = bitsToNum(bits.splice(0, 3))
305+
306+
if (img.lctFlag) {
307+
img.lct = parseCT(1 << (img.lctSize + 1))
308+
}
309+
310+
img.lzwMinCodeSize = st.readByte()
311+
312+
var lzwData = readSubBlocks()
313+
314+
img.pixels = lzwDecode(img.lzwMinCodeSize, lzwData)
315+
316+
if (img.interlaced) {
317+
// Move
318+
img.pixels = deinterlace(img.pixels, img.width)
319+
}
320+
321+
handler.img && handler.img(img)
322+
}
323+
324+
var parseBlock = function () {
325+
var block = {}
326+
block.sentinel = st.readByte()
327+
328+
switch (
329+
String.fromCharCode(block.sentinel) // For ease of matching
330+
) {
331+
case '!':
332+
block.type = 'ext'
333+
parseExt(block)
334+
break
335+
case ',':
336+
block.type = 'img'
337+
parseImg(block)
338+
break
339+
case ';':
340+
block.type = 'eof'
341+
handler.eof && handler.eof(block)
342+
break
343+
default:
344+
throw new Error('Unknown block: 0x' + block.sentinel.toString(16)) // TODO: Pad this with a 0.
345+
}
346+
347+
if (block.type !== 'eof') setTimeout(parseBlock, 0)
348+
}
349+
350+
var parse = function () {
351+
parseHeader()
352+
setTimeout(parseBlock, 0)
353+
}
354+
355+
parse()
356+
}
357+
358+
// BEGIN_NON_BOOKMARKLET_CODE
359+
if (typeof exports !== 'undefined') {
360+
exports.Stream = Stream
361+
exports.parseGIF = parseGIF
362+
}
363+
// export { parseGIF }
364+
// END_NON_BOOKMARKLET_CODE

‎src/utils/file/rgif.js

+124
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,124 @@
1+
import { parseGIF, Stream } from '@/utils/file/gif'
2+
var playGIF = (function () {
3+
var stream
4+
var hdr
5+
6+
var loadError = null
7+
8+
var transparency = null
9+
var delay = null
10+
var disposalMethod = null
11+
var lastDisposalMethod = null
12+
var frame = null
13+
14+
var frames = []
15+
var reset = function () {
16+
transparency = null
17+
delay = null
18+
lastDisposalMethod = disposalMethod
19+
disposalMethod = null
20+
frame = null
21+
frames = []
22+
}
23+
var clear = function () {
24+
transparency = null
25+
delay = null
26+
lastDisposalMethod = disposalMethod
27+
disposalMethod = null
28+
frame = null
29+
}
30+
31+
//gif header info base info
32+
var doHdr = function (_hdr) {
33+
hdr = _hdr
34+
35+
tmpCanvas.width = hdr.width
36+
tmpCanvas.height = hdr.height
37+
}
38+
39+
var doGCE = function (gce) {
40+
pushFrame()
41+
clear()
42+
transparency = gce.transparencyGiven ? gce.transparencyIndex : null
43+
delay = gce.delayTime
44+
disposalMethod = gce.disposalMethod
45+
}
46+
47+
var pushFrame = function () {
48+
if (!frame) return
49+
frames.push({
50+
data: frame.getImageData(0, 0, hdr.width, hdr.height),
51+
delay: delay,
52+
})
53+
}
54+
//图片数据
55+
var doImg = function (img) {
56+
//console.log(img);
57+
if (!frame) frame = tmpCanvas.getContext('2d')
58+
var ct = img.lctFlag ? img.lct : hdr.gct // TODO: What if neither exists?
59+
60+
var cData = frame.getImageData(
61+
img.leftPos,
62+
img.topPos,
63+
img.width,
64+
img.height
65+
)
66+
67+
img.pixels.forEach(function (pixel, i) {
68+
// cData.data === [R,G,B,A,...]
69+
if (transparency !== pixel) {
70+
// This includes null, if no transparency was defined.
71+
cData.data[i * 4 + 0] = ct[pixel][0]
72+
cData.data[i * 4 + 1] = ct[pixel][1]
73+
cData.data[i * 4 + 2] = ct[pixel][2]
74+
cData.data[i * 4 + 3] = 255 // Opaque.
75+
} else {
76+
if (lastDisposalMethod === 2 || lastDisposalMethod === 3) {
77+
cData.data[i * 4 + 3] = 0 // Transparent.
78+
// XXX: This is very very wrong.
79+
} else {
80+
}
81+
}
82+
})
83+
frame.putImageData(cData, img.leftPos, img.topPos)
84+
}
85+
86+
var doNothing = function () {}
87+
88+
var withProgress = function (fn, draw) {
89+
return function (block) {
90+
fn(block)
91+
}
92+
}
93+
94+
var handler = {
95+
hdr: withProgress(doHdr),
96+
gce: withProgress(doGCE),
97+
com: withProgress(doNothing), // I guess that's all for now.
98+
app: {
99+
NETSCAPE: withProgress(doNothing),
100+
},
101+
img: withProgress(doImg),
102+
eof: function (block) {},
103+
}
104+
105+
var tmpCanvas = document.createElement('canvas')
106+
107+
var doParse = function (stream, callback) {
108+
try {
109+
handler.eof = function () {
110+
pushFrame()
111+
callback(hdr, frames, tmpCanvas)
112+
}
113+
reset()
114+
parseGIF(stream, handler)
115+
} catch (err) {
116+
throw new Error('parse error.' + err)
117+
}
118+
}
119+
return {
120+
doParse: doParse,
121+
}
122+
})()
123+
124+
export default playGIF

‎src/views/CloudTest/js/task.js

+24-11
Original file line numberDiff line numberDiff line change
@@ -47,10 +47,12 @@ export default {
4747
original: {},
4848
collectionInfo: {},
4949
drawxnqxPath: '',
50+
profiledrawxnqxPath: '',
5051
thingdata: [],
5152
realtimedata: [],
5253
thingcolumns: [],
5354
historycolumns: [],
55+
historydata: [],
5456
visible: false,
5557
router: '',
5658
topicKey: '',
@@ -219,6 +221,11 @@ export default {
219221
this.$dgiotBus.$on(this.$mqttInfo.topicKey, (mqttMsg) => {
220222
const { data = [] } = JSON.parse(Base64.decode(mqttMsg.payloadString))
221223
this.renderCard(data)
224+
// 刷新存储数据
225+
getDevice(params.objectId).then((res) => {
226+
this.historyEvidence = res.profile.historicaldata
227+
this.drawxnqxPath = res.profile.drawxnqxPath
228+
})
222229
if (data) {
223230
this.collectiontable(data)
224231
// this.renderCard(data)
@@ -402,8 +409,8 @@ export default {
402409
reportId: this.ruleForm.templatenameid,
403410
identifier: 'inspectionReportTemp',
404411
step: 0,
405-
// 0 1 2 3 4
406-
// 开始 取证 完成 生成报告
412+
// 0 1 2 3 4
413+
// 未开始 开始检测中 提交审核 审核完成 生成报告
407414
},
408415
parentId: this.ruleForm.testbedid,
409416
name: this.ruleForm.name,
@@ -449,14 +456,15 @@ export default {
449456
const loading = this.$baseColorfullLoading()
450457
const { results = [] } = await queryEvidence(params)
451458
if (results?.length) {
452-
this.historyEvidence = results[0].original.avgs ?? []
459+
this.historyEvidence = results[0].original.avgs ?? this.historydata
453460
this.collectionInfo.profile.historicaldata = this.historyEvidence
454461
this.historyInfo = results[0]
455-
this.drawxnqxPath = results[0].original.path ?? ''
462+
this.drawxnqxPath =
463+
results[0].original.path ?? this.profiledrawxnqxPath
464+
} else {
465+
this.historyEvidence = this.historydata
466+
this.drawxnqxPath = this.profiledrawxnqxPath
456467
}
457-
this.historycolumns = _.filter(this.thingcolumns, function (item) {
458-
return item.prop !== 'timestamp'
459-
})
460468
// await this.drawxnqx(this.collectionInfo.objectId, this.historyEvidence)
461469
this.$baseMessage(
462470
this.$translateTitle('alert.Data request successfully'),
@@ -823,7 +831,14 @@ export default {
823831
productid: params.parentId.product.objectId,
824832
})
825833
_this.thingcolumns = table
826-
console.log(' _this.thingcolumns', _this.thingcolumns)
834+
_this.historycolumns = _.filter(table, function (item) {
835+
return item.prop !== 'timestamp'
836+
})
837+
if (_this.historycolumns) {
838+
_this.historycolumns = params.profile.historicaldatacolumns
839+
}
840+
_this.historydata = params.profile.historicaldata
841+
_this.profiledrawxnqxPath = params.profile.drawxnqxPath
827842
_this.featHistoryEvidence(this.collectionInfo.objectId)
828843
_this.visible = true
829844
} catch (error) {
@@ -946,12 +961,10 @@ export default {
946961
drawxnqxPath: this.drawxnqxPath,
947962
}),
948963
}
949-
console.log(collectionInfo, _profile)
950964
if (type) this.visible = false
951965
try {
952966
const loading = this.$baseColorfullLoading()
953-
const results = await putDevice(collectionInfo.objectId, _profile)
954-
console.log(results)
967+
await putDevice(collectionInfo.objectId, _profile)
955968
this.$baseMessage(
956969
this.$translateTitle('alert.Data request successfully'),
957970
'success',

1 commit comments

Comments
 (1)

vercel[bot] commented on Aug 12, 2022

@vercel[bot]

Successfully deployed to the following URLs:

dgiot-dashboard – ./

dgiot-dashboard.vercel.app
dgiot-dashboard-git-master-dgiot.vercel.app
dgiot-dashboard-dgiot.vercel.app

Please sign in to comment.