mirror of
https://github.com/CamHenlin/MessagesForMacintosh.git
synced 2024-11-26 05:49:24 +00:00
552 lines
9.3 KiB
JavaScript
552 lines
9.3 KiB
JavaScript
require('cross-fetch/polyfill')
|
|
const ApolloClient = require('apollo-boost').ApolloClient;
|
|
const InMemoryCache = require('apollo-cache-inmemory').InMemoryCache;
|
|
const createHttpLink = require('apollo-link-http').createHttpLink;
|
|
const gql = require('graphql-tag')
|
|
|
|
// TEST_MODE can be turned on or off to prevent communications with the Apollo iMessage Server running on your modern Mac
|
|
const TEST_MODE = false
|
|
|
|
const defaultOptions = {
|
|
watchQuery: {
|
|
fetchPolicy: 'no-cache',
|
|
errorPolicy: 'ignore',
|
|
},
|
|
query: {
|
|
fetchPolicy: 'no-cache',
|
|
errorPolicy: 'ignore',
|
|
},
|
|
}
|
|
|
|
let client
|
|
|
|
const widthFor12ptFont = [
|
|
0,
|
|
10,
|
|
10,
|
|
10,
|
|
10,
|
|
10,
|
|
10,
|
|
10,
|
|
10,
|
|
8,
|
|
10,
|
|
10,
|
|
10,
|
|
0,
|
|
10,
|
|
10,
|
|
10,
|
|
11,
|
|
11,
|
|
9,
|
|
11,
|
|
10,
|
|
10,
|
|
10,
|
|
10,
|
|
10,
|
|
10,
|
|
10,
|
|
10,
|
|
10,
|
|
10,
|
|
10,
|
|
4,
|
|
6,
|
|
7,
|
|
10,
|
|
7,
|
|
11,
|
|
10,
|
|
3,
|
|
5,
|
|
5,
|
|
7,
|
|
7,
|
|
4,
|
|
7,
|
|
4,
|
|
7,
|
|
8,
|
|
8,
|
|
8,
|
|
8,
|
|
8,
|
|
8,
|
|
8,
|
|
8,
|
|
8,
|
|
8,
|
|
4,
|
|
4,
|
|
6,
|
|
8,
|
|
6,
|
|
8,
|
|
11,
|
|
8,
|
|
8,
|
|
8,
|
|
8,
|
|
7,
|
|
7,
|
|
8,
|
|
8,
|
|
6,
|
|
7,
|
|
9,
|
|
7,
|
|
12,
|
|
9,
|
|
8,
|
|
8,
|
|
8,
|
|
8,
|
|
7,
|
|
6,
|
|
8,
|
|
8,
|
|
12,
|
|
8,
|
|
8,
|
|
8,
|
|
5,
|
|
7,
|
|
5,
|
|
8,
|
|
8,
|
|
6,
|
|
8,
|
|
8,
|
|
7,
|
|
8,
|
|
8,
|
|
6,
|
|
8,
|
|
8,
|
|
4,
|
|
6,
|
|
8,
|
|
4,
|
|
12,
|
|
8,
|
|
8,
|
|
8,
|
|
8,
|
|
6,
|
|
7,
|
|
6,
|
|
8,
|
|
8,
|
|
12,
|
|
8,
|
|
8,
|
|
8,
|
|
5,
|
|
5,
|
|
5,
|
|
8,
|
|
8
|
|
]
|
|
|
|
// this is tied to mac_main.c's message window max width
|
|
const MAX_WIDTH = 304
|
|
const SPACE_WIDTH = widthFor12ptFont[32]
|
|
|
|
const getNextWordLength = (word) => {
|
|
|
|
let currentWidth = 0
|
|
|
|
for (const char of word.split(``)) {
|
|
|
|
let currentCharWidth = widthFor12ptFont[char.charCodeAt()]
|
|
|
|
if (isNaN(currentCharWidth)) {
|
|
|
|
currentCharWidth = 1;
|
|
}
|
|
|
|
currentWidth += currentCharWidth
|
|
}
|
|
|
|
return currentWidth
|
|
}
|
|
|
|
const shortenText = (text) => {
|
|
|
|
let outputText = ``
|
|
let currentWidth = 0
|
|
|
|
for (const word of text.split(` `)) {
|
|
|
|
let currentWordWidth = getNextWordLength(word)
|
|
|
|
if (currentWidth + currentWordWidth + SPACE_WIDTH > MAX_WIDTH) {
|
|
|
|
outputText = `${outputText}ENDLASTMESSAGE`
|
|
currentWidth = 0
|
|
|
|
// okay, but what if the word itself is greater than max width?
|
|
if (currentWordWidth > MAX_WIDTH) {
|
|
|
|
let splitWordWidth = 0
|
|
|
|
for (const char of word.split(``)) {
|
|
|
|
let currentCharWidth = widthFor12ptFont[char.charCodeAt()]
|
|
|
|
if (isNaN(currentCharWidth)) {
|
|
|
|
currentCharWidth = 1;
|
|
}
|
|
|
|
if (splitWordWidth + currentCharWidth > MAX_WIDTH) {
|
|
|
|
outputText = `${outputText}ENDLASTMESSAGE`
|
|
splitWordWidth = 0
|
|
}
|
|
|
|
splitWordWidth += currentCharWidth
|
|
outputText = `${outputText}${char}`
|
|
}
|
|
|
|
currentWidth += splitWordWidth
|
|
|
|
continue
|
|
}
|
|
}
|
|
|
|
currentWidth += currentWordWidth + SPACE_WIDTH
|
|
outputText = `${outputText} ${word}`
|
|
}
|
|
|
|
return outputText
|
|
}
|
|
|
|
const MAX_ROWS = 16
|
|
|
|
const splitMessages = (messages) => {
|
|
|
|
let firstMessage = true
|
|
|
|
if (!messages) {
|
|
|
|
return `no messages ENDLASTMESSAGE`
|
|
}
|
|
|
|
for (const message of messages) {
|
|
|
|
if (firstMessage) {
|
|
|
|
let tempMessageOutput = `${message.chatter}: ${message.text}`
|
|
|
|
tempMessageOutput = shortenText(tempMessageOutput)
|
|
messageOutput = tempMessageOutput
|
|
} else {
|
|
|
|
let tempMessageOutput = `${message.chatter}: ${message.text}`
|
|
|
|
tempMessageOutput = shortenText(tempMessageOutput)
|
|
messageOutput = `${messageOutput}ENDLASTMESSAGE${tempMessageOutput}`
|
|
}
|
|
|
|
firstMessage = false
|
|
}
|
|
|
|
|
|
if (messageOutput.split(`ENDLASTMESSAGE`).length > MAX_ROWS) {
|
|
|
|
messageOutput = messageOutput.split(`ENDLASTMESSAGE`)
|
|
|
|
let newMessageOutput = []
|
|
|
|
for (let i = messageOutput.length; i > messageOutput.length - MAX_ROWS; i--) {
|
|
|
|
newMessageOutput.unshift(messageOutput[i])
|
|
}
|
|
|
|
messageOutput = newMessageOutput.join(`ENDLASTMESSAGE`)
|
|
}
|
|
|
|
lastMessageOutput = messageOutput
|
|
|
|
return messageOutput
|
|
}
|
|
|
|
const parseChatsToFriendlyNameString = (chats) => {
|
|
|
|
let friendlyNameStrings = ``
|
|
|
|
if (!chats) {
|
|
|
|
return ``
|
|
}
|
|
|
|
for (let chat of chats) {
|
|
|
|
friendlyNameStrings = `${friendlyNameStrings},${chat.friendlyName.replace(/,/g, '')}`
|
|
}
|
|
|
|
friendlyNameStrings = friendlyNameStrings.substring(1, friendlyNameStrings.length)
|
|
|
|
return friendlyNameStrings
|
|
}
|
|
|
|
let lastMessageOutput
|
|
|
|
let TEST_MESSAGES = [
|
|
{chatter: `friend 1`, text: `my super fun text message`},
|
|
{chatter: `me`, text: `some cool old thing I said earlier`},
|
|
{chatter: `friend 2`, text: `this message is not relevant to the conversation! not at all :(`},
|
|
{chatter: `friend 1`, text: `i watch star wars in reverse order`},
|
|
{chatter: `me`, text: `https://github.com/CamHenlin/MessagesForMacintosh https://github.com/CamHenlin/MessagesForMacintosh`},
|
|
{chatter: `friend 3`, text: `i'm just catching up`},
|
|
{chatter: `friend 3`, text: `nobody chat for a minute`},
|
|
{chatter: `friend 2`, text: `hang on`},
|
|
{chatter: `friend 1`, text: `no`}
|
|
]
|
|
|
|
const TEST_CHATS = [
|
|
{friendlyName: `my group chat 1`, name: `my group chat 1`},
|
|
{friendlyName: `friend 1`, name: `friend 1`},
|
|
{friendlyName: `friend 4`, name: `friend 4`},
|
|
{friendlyName: `boss`, name: `boss`},
|
|
{friendlyName: `friend 3`, name: `friend 3`},
|
|
{friendlyName: `restaurant`, name: `restaurant`}
|
|
]
|
|
|
|
if (TEST_MODE) {
|
|
|
|
setInterval(() => {
|
|
|
|
TEST_MESSAGES = TEST_MESSAGES.concat({chatter: `friend 1`, text: Math.random().toString(36).replace(/[^a-z]+/g, '').substr(0, 64)})
|
|
}, 10000)
|
|
}
|
|
|
|
class iMessageClient {
|
|
|
|
async getMessages (chatId, page) {
|
|
|
|
if (TEST_MODE) {
|
|
|
|
return splitMessages(TEST_MESSAGES)
|
|
}
|
|
|
|
console.log(`get messages for chat ID: ${chatId}`)
|
|
|
|
let result
|
|
|
|
try {
|
|
|
|
result = await client.query({
|
|
query: gql`query getMessages {
|
|
getMessages(chatId: "${chatId}", page: "${page}") {
|
|
chatter
|
|
text
|
|
}
|
|
}`
|
|
})
|
|
} catch (error) {
|
|
|
|
console.log(`error with apollo query`)
|
|
console.log(error)
|
|
|
|
result = {
|
|
data: {
|
|
}
|
|
}
|
|
}
|
|
|
|
let messages = result.data.getMessages
|
|
|
|
return splitMessages(messages)
|
|
}
|
|
|
|
async hasNewMessagesInChat (chatId) {
|
|
|
|
let currentLastMessageOutput = `${lastMessageOutput}`
|
|
let messageOutput = await this.getMessages(chatId, 0)
|
|
|
|
return (currentLastMessageOutput !== messageOutput).toString()
|
|
}
|
|
|
|
async hasNewMessages (chatId) {
|
|
|
|
let currentLastMessageOutput = `${lastMessageOutput}`
|
|
let messageOutput = await this.getChats(chatId, 0)
|
|
|
|
return (currentLastMessageOutput !== messageOutput).toString()
|
|
}
|
|
|
|
async sendMessage (chatId, message) {
|
|
|
|
if (TEST_MODE) {
|
|
|
|
TEST_MESSAGES = TEST_MESSAGES.concat({chatter: `me`, text: message})
|
|
|
|
return splitMessages(TEST_MESSAGES)
|
|
}
|
|
|
|
let result
|
|
|
|
try {
|
|
|
|
result = await client.query({
|
|
query: gql`query sendMessage {
|
|
sendMessage(chatId: "${chatId}", message: "${message}") {
|
|
chatter
|
|
text
|
|
}
|
|
}`
|
|
})
|
|
} catch (error) {
|
|
|
|
console.log(`error with apollo query`)
|
|
console.log(error)
|
|
|
|
result = {
|
|
data: {
|
|
}
|
|
}
|
|
}
|
|
|
|
let messages = result.data.sendMessage
|
|
|
|
return splitMessages(messages)
|
|
}
|
|
|
|
async getChats () {
|
|
console.log(`getChats`)
|
|
|
|
if (TEST_MODE) {
|
|
|
|
return parseChatsToFriendlyNameString(TEST_CHATS)
|
|
}
|
|
|
|
let result
|
|
|
|
try {
|
|
|
|
result = await client.query({
|
|
query: gql`query getChats {
|
|
getChats {
|
|
name
|
|
friendlyName
|
|
}
|
|
}`
|
|
})
|
|
} catch (error) {
|
|
|
|
console.log(`error with apollo query`)
|
|
console.log(error)
|
|
|
|
result = {
|
|
data: {
|
|
}
|
|
}
|
|
}
|
|
|
|
let chats = result.data.getChats
|
|
|
|
console.log(`getChats complete`)
|
|
|
|
return parseChatsToFriendlyNameString(chats)
|
|
}
|
|
|
|
async getChatCounts () {
|
|
|
|
console.log(`getChatCounts`)
|
|
|
|
if (TEST_MODE) {
|
|
|
|
return parseChatsToFriendlyNameString(TEST_CHATS)
|
|
}
|
|
|
|
let result
|
|
|
|
try {
|
|
|
|
result = await client.query({
|
|
query: gql`query getChatCounts {
|
|
getChatCounts {
|
|
friendlyName
|
|
count
|
|
}
|
|
}`
|
|
})
|
|
} catch (error) {
|
|
|
|
console.log(`error with apollo query`)
|
|
console.log(error)
|
|
|
|
result = {
|
|
data: {
|
|
}
|
|
}
|
|
}
|
|
|
|
let chats = result.data.getChatCounts
|
|
|
|
console.log(`got chat counts`)
|
|
|
|
if (!chats) {
|
|
|
|
return ``
|
|
}
|
|
|
|
let friendlyNameStrings = ``
|
|
|
|
if (chats.length === 0) {
|
|
|
|
return ``
|
|
}
|
|
|
|
for (let chat of chats) {
|
|
|
|
friendlyNameStrings = `${friendlyNameStrings},${chat.friendlyName.replace(/,/g, '')}:::${chat.count}`
|
|
}
|
|
|
|
// remove trailing comma
|
|
friendlyNameStrings = friendlyNameStrings.substring(1, friendlyNameStrings.length)
|
|
|
|
console.log(friendlyNameStrings)
|
|
return friendlyNameStrings
|
|
}
|
|
|
|
setIPAddress (IPAddress) {
|
|
|
|
console.log(`instantiate apolloclient with uri ${IPAddress}:4000/`)
|
|
|
|
if (TEST_MODE) {
|
|
|
|
return `success`
|
|
}
|
|
|
|
try {
|
|
|
|
client = new ApolloClient({
|
|
uri: `${IPAddress}:4000/`,
|
|
cache: new InMemoryCache(),
|
|
link: new createHttpLink({
|
|
uri: `${IPAddress}:4000/`
|
|
}),
|
|
defaultOptions
|
|
});
|
|
} catch (err) {
|
|
console.log(`error instantiating the ApolloClient`)
|
|
console.log(err)
|
|
|
|
return `failure`
|
|
}
|
|
|
|
console.log(`return success`)
|
|
|
|
return `success`
|
|
}
|
|
}
|
|
|
|
module.exports = iMessageClient
|
|
|