1
0
Fork 0
BangleApps/apps/followtherecipe/app.js

226 lines
7.6 KiB
JavaScript

const settings = require("Storage").readJSON("followtherecipe.json");
//const locale = require('locale');
var W = g.getWidth(), H = g.getHeight();
var screen = 0;
let maxLenghtHorizontal = 16;
let maxLenghtvertical = 6;
let uri = "https://www.themealdb.com/api/json/v1/1/random.php";
var colors = {0: "#70f", 1:"#70d", 2: "#70g", 3: "#20f", 4: "#30f"};
var screens = [];
function drawData(name, value, y){
g.drawString(name, 10, y);
g.drawString(value, 100, y);
}
function drawInfo() {
g.reset().clearRect(Bangle.appRect);
var h=18, y = h;
// Header
g.drawLine(0,25,W,25);
g.drawLine(0,26,W,26);
// Info body depending on screen
g.setFont("Vector",15).setFontAlign(-1,-1).setColor("#0ff");
screens[screen].items.forEach(function (item, index){
g.setColor(colors[index]);
drawData(item.name, item.fun, y+=h);
});
// Bottom
g.setColor(g.theme.fg);
g.drawLine(0,H-h-3,W,H-h-3);
g.drawLine(0,H-h-2,W,H-h-2);
g.setFont("Vector",h-2).setFontAlign(-1,-1);
g.drawString(screens[screen].name, 2, H-h+2);
g.setFont("Vector",h-2).setFontAlign(1,-1);
g.drawString((screen+1) + "/" + screens.length, W, H-h+2);
}
// Change page if user touch the left or the right of the screen
Bangle.on('touch', function(btn, e){
var left = parseInt(g.getWidth() * 0.3);
var right = g.getWidth() - left;
var isLeft = e.x < left;
var isRight = e.x > right;
if(isRight){
screen = (screen + 1) % screens.length;
}
if(isLeft){
screen -= 1;
screen = screen < 0 ? screens.length-1 : screen;
}
Bangle.buzz(40, 0.6);
drawInfo();
});
function infoIngredients(ingredients, measures){
let combinedList = [];
let listOfString = [];
let lineBreaks = 0;
// Iterate through the arrays and combine the ingredients and measures
for (let i = 0; i < ingredients.length; i++) {
const combinedString = `${ingredients[i]}: ${measures[i]}`;
lineBreaks += 1;
// Check if the line is more than 16 characters
if (combinedString.length > maxLenghtHorizontal) {
// Add line break and update lineBreaks counter
combinedList.push(`${ingredients[i]}:\n${measures[i]}`);
lineBreaks += 1;
} else {
// Add to the combinedList array
combinedList.push(combinedString);
}
// Check the total line breaks
if (lineBreaks >= maxLenghtvertical) {
const resultString = combinedList.join('\n');
listOfString.push(resultString);
combinedList = [];
lineBreaks = 0;
}
if(i == ingredients.length){
listOfString.push(combinedList.join('\n'));
}
}
for(let i = 0; i < listOfString.length; i++){
let screen = {
name: "Ingredients",
items: [
{name: listOfString[i], fun: ""},
]
};
screens.push(screen);
}
}
// Format instructions to display on screen
function infoInstructions(instructionsString){
let item = [];
let chunkSize = 22;
//remove all space line and other to avoid problem with text
instructionsString = instructionsString.replace(/[\n\r]/g, '');
for (let i = 0; i < instructionsString.length; i += chunkSize) {
const chunk = instructionsString.substring(i, i + chunkSize).trim();
item.push({ name: chunk, fun: "" });
if (item.length === maxLenghtvertical) {
let screen = {
name: "Instructions",
items: item,
};
screens.push(screen);
item = [];
}
}
if (item.length > 0) {
let screen = {
name: "Instructions",
items: item,
};
screens.push(screen);
}
}
// Get json format and parse it into Strings
function getRecipeData(data) {
let mealName = data.strMeal;
let category = data.strCategory;
let area = data.strArea;
let instructions = data.strInstructions;
const ingredients = [];
const measures = [];
for (let i = 1; i <= 20; i++) {
const ingredient = data["strIngredient" + i];
const measure = data["strMeasure" + i];
if (ingredient && ingredient.trim() !== "") {
ingredients.push(ingredient);
if (measure && measure.trim() !== ""){
measures.push(measure);
}else{
measures.push("¯\\_(ツ)_/¯");
}
} else { // If no more ingredients are found
screens = [
{
name: "General",
items: [
{name: mealName, fun: ""},
{name: "", fun: ""},
{name: "Category", fun: category},
{name: "", fun: ""},
{name: "Area: ", fun: area},
]
}
];
infoIngredients(ingredients, measures);
infoInstructions(instructions);
drawInfo();
break;
}
}
}
function jsonData(){
let json = '{"meals":[{"idMeal":"52771","strMeal":"Spicy Arrabiata Penne","strDrinkAlternate":null,"strCategory":"Vegetarian","strArea":"Italian","strInstructions":"Bring a large pot of water to a boil. Add kosher salt to the boiling water, then add the pasta. Cook according to the package instructions, about 9 minutes.\\r\\nIn a large skillet over medium-high heat, add the olive oil and heat until the oil starts to shimmer. Add the garlic and cook, stirring, until fragrant, 1 to 2 minutes. Add the chopped tomatoes, red chile flakes, Italian seasoning and salt and pepper to taste. Bring to a boil and cook for 5 minutes. Remove from the heat and add the chopped basil.\\r\\nDrain the pasta and add it to the sauce. Garnish with Parmigiano-Reggiano flakes and more basil and serve warm.","strMealThumb":"https://www.themealdb.com/images/media/meals/ustsqw1468250014.jpg","strTags":"Pasta,Curry","strYoutube":"https://www.youtube.com/watch?v=1IszT_guI08","strIngredient1":"penne rigate","strIngredient2":"olive oil","strIngredient3":"garlic","strIngredient4":"chopped tomatoes","strIngredient5":"red chile flakes","strIngredient6":"italian seasoning","strIngredient7":"basil","strIngredient8":"Parmigiano-Reggiano","strIngredient9":"","strIngredient10":"","strIngredient11":"","strIngredient12":"","strIngredient13":"","strIngredient14":"","strIngredient15":"","strIngredient16":null,"strIngredient17":null,"strIngredient18":null,"strIngredient19":null,"strIngredient20":null,"strMeasure1":"1 pound","strMeasure2":"1/4 cup","strMeasure3":"3 cloves","strMeasure4":"1 tin ","strMeasure5":"1/2 teaspoon","strMeasure6":"1/2 teaspoon","strMeasure7":"6 leaves","strMeasure8":"spinkling","strMeasure9":"","strMeasure10":"","strMeasure11":"","strMeasure12":"","strMeasure13":"","strMeasure14":"","strMeasure15":"","strMeasure16":null,"strMeasure17":null,"strMeasure18":null,"strMeasure19":null,"strMeasure20":null,"strSource":null,"strImageSource":null,"strCreativeCommonsConfirmed":null,"dateModified":null}]}';
if(settings != null){
json = JSON.stringify({ meals: [settings] });
}
const obj = JSON.parse(json);
getRecipeData(obj.meals[0]);
}
function initData(retryCount) {
if (!Bangle.http) {
console.log("No http method found");
jsonData();
return;
}
jsonData();
Bangle.http(uri, { timeout: 1000 })
.then(event => {
try {
const obj = JSON.parse(event.resp);
if (obj.meals && obj.meals.length > 0) {
getRecipeData(obj.meals[0]);
} else {
console.log("Invalid JSON structure: meals array is missing or empty");
}
} catch (error) {
console.log("JSON Parse Error: " + error.message);
}
})
.catch(e => {
console.log("Request Error:", e);
if (e === "Timeout" && retryCount > 0) {
setTimeout(() => initData(retryCount - 1), 1000); // Optional: Add a delay before retrying
}else{
jsonData();
}
});
}
initData(3);
Bangle.on('lock', function(isLocked) {
drawInfo();
});
Bangle.loadWidgets();
Bangle.drawWidgets();