Compare commits
No commits in common. "f75001d03b6983171c23e94a8fbdfc6071653972" and "e5216b24e13d8aea4ea0fef18beb4de4e36be3b8" have entirely different histories.
f75001d03b
...
e5216b24e1
38
index.html
38
index.html
|
@ -4,38 +4,19 @@
|
||||||
<meta charset="UTF-8">
|
<meta charset="UTF-8">
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
<title>Calendar Merger</title>
|
<title>Calendar Merger</title>
|
||||||
<style>
|
<link rel="stylesheet" href="/styles.css">
|
||||||
body {
|
|
||||||
font-family: Arial, sans-serif;
|
|
||||||
}
|
|
||||||
#calendars {
|
|
||||||
margin-bottom: 20px;
|
|
||||||
}
|
|
||||||
#calendars .calendar {
|
|
||||||
margin-bottom: 10px;
|
|
||||||
}
|
|
||||||
#calendars .calendar input[type="text"] {
|
|
||||||
width: 50%;
|
|
||||||
margin-right: 10px;
|
|
||||||
}
|
|
||||||
#calendars .calendar input[type="url"] {
|
|
||||||
width: 50%;
|
|
||||||
}
|
|
||||||
#add-calendar {
|
|
||||||
margin-bottom: 20px;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<h1>Calendar Merger</h1>
|
<h1>Calendar Merger</h1>
|
||||||
<form id="merge-form">
|
<form id="merge-form">
|
||||||
<div id="calendars">
|
<label for="cal1-url">Calendar 1 URL:</label>
|
||||||
<div class="calendar">
|
<input type="url" id="cal1-url" name="cal1-url"><br><br>
|
||||||
<input type="text" id="prefix-0" placeholder="Prefix">
|
<label for="cal1-prefix">Calendar 1 Prefix:</label>
|
||||||
<input type="url" id="url-0" placeholder="Calendar URL">
|
<input type="text" id="cal1-prefix" name="cal1-prefix"><br><br>
|
||||||
</div>
|
<label for="cal2-url">Calendar 2 URL:</label>
|
||||||
</div>
|
<input type="url" id="cal2-url" name="cal2-url"><br><br>
|
||||||
<button id="add-calendar" type="button">Add Calendar</button>
|
<label for="cal2-prefix">Calendar 2 Prefix:</label>
|
||||||
|
<input type="text" id="cal2-prefix" name="cal2-prefix"><br><br>
|
||||||
<button type="submit">Merge Calendars</button>
|
<button type="submit">Merge Calendars</button>
|
||||||
</form>
|
</form>
|
||||||
<div id="result"></div>
|
<div id="result"></div>
|
||||||
|
@ -43,4 +24,3 @@
|
||||||
<script src="script.js"></script>
|
<script src="script.js"></script>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
|
||||||
|
|
45
script.js
45
script.js
|
@ -1,42 +1,31 @@
|
||||||
const form = document.getElementById('merge-form');
|
const form = document.getElementById('merge-form');
|
||||||
const calendars = document.getElementById('calendars');
|
const resultDiv = document.getElementById('result');
|
||||||
const addCalendarButton = document.getElementById('add-calendar');
|
|
||||||
const result = document.getElementById('result');
|
|
||||||
|
|
||||||
let calendarIndex = 1;
|
form.addEventListener('submit', (e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
const cal1Url = document.getElementById('cal1-url').value;
|
||||||
|
const cal1Prefix = document.getElementById('cal1-prefix').value;
|
||||||
|
const cal2Url = document.getElementById('cal2-url').value;
|
||||||
|
const cal2Prefix = document.getElementById('cal2-prefix').value;
|
||||||
|
|
||||||
addCalendarButton.addEventListener('click', () => {
|
|
||||||
const newCalendar = document.createElement('div');
|
|
||||||
newCalendar.className = 'calendar';
|
|
||||||
newCalendar.innerHTML = `
|
|
||||||
<input type="text" id="prefix-${calendarIndex}" placeholder="Prefix">
|
|
||||||
<input type="url" id="url-${calendarIndex}" placeholder="Calendar URL">
|
|
||||||
`;
|
|
||||||
calendars.appendChild(newCalendar);
|
|
||||||
calendarIndex++;
|
|
||||||
});
|
|
||||||
|
|
||||||
form.addEventListener('submit', (event) => {
|
|
||||||
event.preventDefault();
|
|
||||||
const calendarsData = [];
|
|
||||||
for (let i = 0; i < calendarIndex; i++) {
|
|
||||||
const prefix = document.getElementById(`prefix-${i}`).value;
|
|
||||||
const url = document.getElementById(`url-${i}`).value;
|
|
||||||
calendarsData.push({ prefix, url });
|
|
||||||
}
|
|
||||||
fetch('/merge', {
|
fetch('/merge', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: {
|
headers: {
|
||||||
'Content-Type': 'application/json'
|
'Content-Type': 'application/json'
|
||||||
},
|
},
|
||||||
body: JSON.stringify({ calendars: calendarsData })
|
body: JSON.stringify({
|
||||||
|
cal1Url,
|
||||||
|
cal1Prefix,
|
||||||
|
cal2Url,
|
||||||
|
cal2Prefix
|
||||||
})
|
})
|
||||||
.then((response) => response.json())
|
})
|
||||||
|
.then(response => response.json())
|
||||||
.then((data) => {
|
.then((data) => {
|
||||||
result.innerHTML = `Merged calendar URL: <a href="${data.url}">${data.url}</a>`;
|
resultDiv.innerHTML = `Merged calendar URL: <a href="${data.url}" target="_blank">${data.url}</a>`;
|
||||||
})
|
})
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
result.innerHTML = 'Error merging calendars';
|
resultDiv.innerHTML = 'Error merging calendars';
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
});
|
47
server.js
47
server.js
|
@ -14,46 +14,36 @@ app.get('/', (req, res) => {
|
||||||
});
|
});
|
||||||
|
|
||||||
app.post('/merge', async (req, res) => {
|
app.post('/merge', async (req, res) => {
|
||||||
const { calendars } = req.body;
|
const { cal1Url, cal1Prefix, cal2Url, cal2Prefix } = req.body;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
//validate the input
|
|
||||||
if (!calendars || !Array.isArray(calendars)) {
|
|
||||||
return res.status(400).json({ error: 'Invalid input' });
|
|
||||||
}
|
|
||||||
// Fetch calendar data from URLs
|
// Fetch calendar data from URLs
|
||||||
const promises = calendars.map((calendar) => {
|
const cal1Data = await axios.get(cal1Url);
|
||||||
return axios.get(calendar.url)
|
const cal2Data = await axios.get(cal2Url);
|
||||||
.then((response) => {
|
|
||||||
return {
|
|
||||||
data: response.data,
|
|
||||||
prefix: calendar.prefix,
|
|
||||||
};
|
|
||||||
})
|
|
||||||
.catch((error) => {
|
|
||||||
console.error(error);
|
|
||||||
return null;
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
const results = await Promise.all(promises);
|
|
||||||
// Filter out any failed requests
|
|
||||||
const validResults = results.filter((result) => result !== null);
|
|
||||||
|
|
||||||
// Parse calendar data
|
// Parse calendar data
|
||||||
|
const cal1 = ical.parseICS(cal1Data.data);
|
||||||
|
const cal2 = ical.parseICS(cal2Data.data);
|
||||||
const mergedCal = [];
|
const mergedCal = [];
|
||||||
validResults.forEach((result) => {
|
|
||||||
const calendar = ical.parseICS(result.data);
|
// Merge calendars
|
||||||
Object.keys(calendar).forEach((key) => {
|
Object.keys(cal1).forEach((key) => {
|
||||||
const event = calendar[key];
|
let event = cal1[key];
|
||||||
mergedCal.push({
|
mergedCal.push({
|
||||||
start: event.start,
|
start: event.start,
|
||||||
end: event.end,
|
end: event.end,
|
||||||
summary: `${result.prefix} ${event.summary}`,
|
summary: `${cal1Prefix} ${event.summary}`,
|
||||||
});
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
Object.keys(cal2).forEach((key) => {
|
||||||
|
let event = cal2[key];
|
||||||
|
mergedCal.push({
|
||||||
|
start: event.start,
|
||||||
|
end: event.end,
|
||||||
|
summary: `${cal2Prefix} ${event.summary}`,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
// Save merged calendar to file
|
// Save merged calendar to file
|
||||||
const filename = `merged-${Date.now()}.ics`;
|
const filename = `merged-${Date.now()}.ics`;
|
||||||
|
@ -73,7 +63,6 @@ END:VEVENT
|
||||||
icalString += `END:VCALENDAR`;
|
icalString += `END:VCALENDAR`;
|
||||||
fs.writeFileSync(filename, icalString);
|
fs.writeFileSync(filename, icalString);
|
||||||
|
|
||||||
|
|
||||||
// Generate a unique URL for the merged calendar
|
// Generate a unique URL for the merged calendar
|
||||||
const mergedCalendarUrl = `${req.protocol}://${req.get('host')}/${filename}`;
|
const mergedCalendarUrl = `${req.protocol}://${req.get('host')}/${filename}`;
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue