'use strict'; // Exposes the internal Store to the WhatsApp Web client exports.ExposeStore = (moduleRaidStr) => { eval('var moduleRaid = ' + moduleRaidStr); // eslint-disable-next-line no-undef window.mR = moduleRaid(); window.Store = window.mR.findModule('Chat')[0].default; window.Store.AppState = window.mR.findModule('STREAM')[0].default; window.Store.Conn = window.mR.findModule('Conn')[0].default; window.Store.CryptoLib = window.mR.findModule('decryptE2EMedia')[0]; window.Store.Wap = window.mR.findModule('Wap')[0].default; window.Store.SendSeen = window.mR.findModule('sendSeen')[0]; window.Store.SendClear = window.mR.findModule('sendClear')[0]; window.Store.SendDelete = window.mR.findModule('sendDelete')[0]; window.Store.genId = window.mR.findModule((module) => module.default && typeof module.default === 'function' && module.default.toString().match(/crypto/))[0].default; window.Store.SendMessage = window.mR.findModule('addAndSendMsgToChat')[0]; window.Store.MsgKey = window.mR.findModule((module) => module.default && module.default.fromString)[0].default; window.Store.Invite = window.mR.findModule('sendJoinGroupViaInvite')[0]; window.Store.OpaqueData = window.mR.findModule(module => module.default && module.default.createFromData)[0].default; window.Store.MediaPrep = window.mR.findModule('MediaPrep')[0]; window.Store.MediaObject = window.mR.findModule('getOrCreateMediaObject')[0]; window.Store.MediaUpload = window.mR.findModule('uploadMedia')[0]; window.Store.Cmd = window.mR.findModule('Cmd')[0].default; window.Store.MediaTypes = window.mR.findModule('msgToMediaType')[0]; window.Store.VCard = window.mR.findModule('vcardFromContactModel')[0]; window.Store.UserConstructor = window.mR.findModule((module) => (module.default && module.default.prototype && module.default.prototype.isServer && module.default.prototype.isUser) ? module.default : null)[0].default; window.Store.Validators = window.mR.findModule('findLinks')[0]; window.Store.WidFactory = window.mR.findModule('createWid')[0]; window.Store.BlockContact = window.mR.findModule('blockContact')[0]; window.Store.GroupMetadata = window.mR.findModule((module) => module.default && module.default.handlePendingInvite)[0].default; window.Store.Sticker = window.mR.findModule('Sticker')[0].default.Sticker; window.Store.UploadUtils = window.mR.findModule((module) => (module.default && module.default.encryptAndUpload) ? module.default : null)[0].default; window.Store.Label = window.mR.findModule('LabelCollection')[0].default; }; exports.LoadUtils = () => { window.WWebJS = {}; window.WWebJS.getNumberId = async (id) => { let result = await window.Store.Wap.queryExist(id); if (result.jid === undefined) throw 'The number provided is not a registered whatsapp user'; return result.jid; }; window.WWebJS.sendSeen = async (chatId) => { let chat = window.Store.Chat.get(chatId); if (chat !== undefined) { await window.Store.SendSeen.sendSeen(chat, false); return true; } return false; }; window.WWebJS.sendMessage = async (chat, content, options = {}) => { let attOptions = {}; if (options.attachment) { attOptions = options.sendMediaAsSticker ? await window.WWebJS.processStickerData(options.attachment) : await window.WWebJS.processMediaData(options.attachment, { forceVoice: options.sendAudioAsVoice, forceDocument: options.sendMediaAsDocument }); content = options.sendMediaAsSticker ? undefined : attOptions.preview; delete options.attachment; delete options.sendMediaAsSticker; } let quotedMsgOptions = {}; if (options.quotedMessageId) { let quotedMessage = window.Store.Msg.get(options.quotedMessageId); if (quotedMessage.canReply()) { quotedMsgOptions = quotedMessage.msgContextInfo(chat); } delete options.quotedMessageId; } if (options.mentionedJidList) { options.mentionedJidList = options.mentionedJidList.map(cId => window.Store.Contact.get(cId).id); } let locationOptions = {}; if (options.location) { locationOptions = { type: 'location', loc: options.location.description, lat: options.location.latitude, lng: options.location.longitude }; delete options.location; } let vcardOptions = {}; if (options.contactCard) { let contact = window.Store.Contact.get(options.contactCard); vcardOptions = { body: window.Store.VCard.vcardFromContactModel(contact).vcard, type: 'vcard', vcardFormattedName: contact.formattedName }; delete options.contactCard; } else if (options.contactCardList) { let contacts = options.contactCardList.map(c => window.Store.Contact.get(c)); let vcards = contacts.map(c => window.Store.VCard.vcardFromContactModel(c)); vcardOptions = { type: 'multi_vcard', vcardList: vcards, body: undefined }; delete options.contactCardList; } else if (options.parseVCards && typeof (content) === 'string' && content.startsWith('BEGIN:VCARD')) { delete options.parseVCards; try { const parsed = window.Store.VCard.parseVcard(content); if (parsed) { vcardOptions = { type: 'vcard', vcardFormattedName: window.Store.VCard.vcardGetNameFromParsed(parsed) }; } } catch (_) { // not a vcard } } if (options.linkPreview) { delete options.linkPreview; const link = window.Store.Validators.findLink(content); if (link) { const preview = await window.Store.Wap.queryLinkPreview(link.url); preview.preview = true; preview.subtype = 'url'; options = { ...options, ...preview }; } } const newMsgId = new window.Store.MsgKey({ fromMe: true, remote: chat.id, id: window.Store.genId(), }); const message = { ...options, id: newMsgId, ack: 0, body: content, from: window.Store.Conn.wid, to: chat.id, local: true, self: 'out', t: parseInt(new Date().getTime() / 1000), isNewMsg: true, type: 'chat', ...locationOptions, ...attOptions, ...quotedMsgOptions, ...vcardOptions }; await window.Store.SendMessage.addAndSendMsgToChat(chat, message); return window.Store.Msg.get(newMsgId._serialized); }; window.WWebJS.processStickerData = async (mediaInfo) => { if (mediaInfo.mimetype !== 'image/webp') throw new Error('Invalid media type'); const file = window.WWebJS.mediaInfoToFile(mediaInfo); let filehash = await window.WWebJS.getFileHash(file); let mediaKey = await window.WWebJS.generateHash(32); const controller = new AbortController(); const uploadedInfo = await window.Store.UploadUtils.encryptAndUpload({ blob: file, type: 'sticker', signal: controller.signal, mediaKey }); const stickerInfo = { ...uploadedInfo, clientUrl: uploadedInfo.url, uploadhash: uploadedInfo.encFilehash, size: file.size, type: 'sticker', filehash }; return stickerInfo; }; window.WWebJS.processMediaData = async (mediaInfo, { forceVoice, forceDocument }) => { const file = window.WWebJS.mediaInfoToFile(mediaInfo); const mData = await window.Store.OpaqueData.createFromData(file, file.type); const mediaPrep = window.Store.MediaPrep.prepRawMedia(mData, { asDocument: forceDocument }); const mediaData = await mediaPrep.waitForPrep(); const mediaObject = window.Store.MediaObject.getOrCreateMediaObject(mediaData.filehash); const mediaType = window.Store.MediaTypes.msgToMediaType({ type: mediaData.type, isGif: mediaData.isGif }); if (forceVoice && mediaData.type === 'audio') { mediaData.type = 'ptt'; } if (forceDocument) { mediaData.type = 'document'; } if (!(mediaData.mediaBlob instanceof window.Store.OpaqueData)) { mediaData.mediaBlob = await window.Store.OpaqueData.createFromData(mediaData.mediaBlob, mediaData.mediaBlob.type); } mediaData.renderableUrl = mediaData.mediaBlob.url(); mediaObject.consolidate(mediaData.toJSON()); mediaData.mediaBlob.autorelease(); const uploadedMedia = await window.Store.MediaUpload.uploadMedia({ mimetype: mediaData.mimetype, mediaObject, mediaType }); const mediaEntry = uploadedMedia.mediaEntry; if (!mediaEntry) { throw new Error('upload failed: media entry was not created'); } mediaData.set({ clientUrl: mediaEntry.mmsUrl, directPath: mediaEntry.directPath, mediaKey: mediaEntry.mediaKey, mediaKeyTimestamp: mediaEntry.mediaKeyTimestamp, filehash: mediaObject.filehash, uploadhash: mediaEntry.uploadHash, size: mediaObject.size, streamingSidecar: mediaEntry.sidecar, firstFrameSidecar: mediaEntry.firstFrameSidecar }); return mediaData; }; window.WWebJS.getMessageModel = message => { const msg = message.serialize(); msg.isStatusV3 = message.isStatusV3; msg.links = (message.getLinks()).map(link => link.href); if (msg.buttons) { msg.buttons = msg.buttons.serialize(); } delete msg.pendingAckUpdate; return msg; }; window.WWebJS.getChatModel = async chat => { let res = chat.serialize(); res.isGroup = chat.isGroup; res.formattedTitle = chat.formattedTitle; res.isMuted = chat.mute && chat.mute.isMuted; if (chat.groupMetadata) { await window.Store.GroupMetadata.update(chat.id._serialized); res.groupMetadata = chat.groupMetadata.serialize(); } delete res.msgs; delete res.msgUnsyncedButtonReplyMsgs; delete res.unsyncedButtonReplies; return res; }; window.WWebJS.getChat = async chatId => { const chat = window.Store.Chat.get(chatId); return await window.WWebJS.getChatModel(chat); }; window.WWebJS.getChats = async () => { const chats = window.Store.Chat.models; const chatPromises = chats.map(chat => window.WWebJS.getChatModel(chat)); return await Promise.all(chatPromises); }; window.WWebJS.getContactModel = contact => { let res = contact.serialize(); res.isBusiness = contact.isBusiness; if (contact.businessProfile) { res.businessProfile = contact.businessProfile.serialize(); } res.isMe = contact.isMe; res.isUser = contact.isUser; res.isGroup = contact.isGroup; res.isWAContact = contact.isWAContact; res.isMyContact = contact.isMyContact; res.isBlocked = contact.isContactBlocked; res.userid = contact.userid; return res; }; window.WWebJS.getContact = contactId => { const contact = window.Store.Contact.get(contactId); return window.WWebJS.getContactModel(contact); }; window.WWebJS.getContacts = () => { const contacts = window.Store.Contact.models; return contacts.map(contact => window.WWebJS.getContactModel(contact)); }; window.WWebJS.mediaInfoToFile = ({ data, mimetype, filename }) => { const binaryData = atob(data); const buffer = new ArrayBuffer(binaryData.length); const view = new Uint8Array(buffer); for (let i = 0; i < binaryData.length; i++) { view[i] = binaryData.charCodeAt(i); } const blob = new Blob([buffer], { type: mimetype }); return new File([blob], filename, { type: mimetype, lastModified: Date.now() }); }; window.WWebJS.downloadBuffer = (url) => { return new Promise(function (resolve, reject) { let xhr = new XMLHttpRequest(); xhr.open('GET', url); xhr.responseType = 'arraybuffer'; xhr.onload = function () { if (xhr.status == 200) { resolve(xhr.response); } else { reject({ status: this.status, statusText: xhr.statusText }); } }; xhr.onerror = function () { reject({ status: this.status, statusText: xhr.statusText }); }; xhr.send(null); }); }; window.WWebJS.readBlobAsync = (blob) => { return new Promise((resolve, reject) => { let reader = new FileReader(); reader.onload = () => { resolve(reader.result); }; reader.onerror = reject; reader.readAsDataURL(blob); }); }; window.WWebJS.getFileHash = async (data) => { let buffer = await data.arrayBuffer(); const hashBuffer = await crypto.subtle.digest('SHA-256', buffer); return btoa(String.fromCharCode(...new Uint8Array(hashBuffer))); }; window.WWebJS.generateHash = async (length) => { var result = ''; var characters = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789'; var charactersLength = characters.length; for ( var i = 0; i < length; i++ ) { result += characters.charAt(Math.floor(Math.random() * charactersLength)); } return result; }; window.WWebJS.sendClearChat = async (chatId) => { let chat = window.Store.Chat.get(chatId); if (chat !== undefined) { await window.Store.SendClear.sendClear(chat, false); return true; } return false; }; window.WWebJS.sendDeleteChat = async (chatId) => { let chat = window.Store.Chat.get(chatId); if (chat !== undefined) { await window.Store.SendDelete.sendDelete(chat); return true; } return false; }; window.WWebJS.sendChatstate = async (state, chatId) => { switch (state) { case 'typing': await window.Store.Wap.sendChatstateComposing(chatId); break; case 'recording': await window.Store.Wap.sendChatstateRecording(chatId); break; case 'stop': await window.Store.Wap.sendChatstatePaused(chatId); break; default: throw 'Invalid chatstate'; } return true; }; window.WWebJS.getLabelModel = label => { let res = label.serialize(); res.hexColor = label.hexColor; return res; }; window.WWebJS.getLabels = () => { const labels = window.Store.Label.models; return labels.map(label => window.WWebJS.getLabelModel(label)); }; window.WWebJS.getLabel = (labelId) => { const label = window.Store.Label.get(labelId); return window.WWebJS.getLabelModel(label); }; window.WWebJS.getChatLabels = async (chatId) => { const chat = await window.WWebJS.getChat(chatId); return (chat.labels || []).map(id => window.WWebJS.getLabel(id)); }; }; exports.MarkAllRead = () => { let Chats = window.Store.Chat.models; for (let chatIndex in Chats) { if (isNaN(chatIndex)) { continue; } let chat = Chats[chatIndex]; if (chat.unreadCount > 0) { chat.markSeen(); window.Store.Wap.sendConversationSeen(chat.id, chat.getLastMsgKeyForAction(), chat.unreadCount - chat.pendingSeenCount); } } };