/** ****************************************************************************** * @addtogroup OpenPilotModules OpenPilot Modules * @{ * @addtogroup OSDgenModule osdgen Module * @brief Process OSD information * @{ * * @file osdgen.c * @author The OpenPilot Team, http://www.openpilot.org Copyright (C) 2010. * @brief OSD gen module, handles OSD draw. Parts from CL-OSD and SUPEROSD projects * @see The GNU Public License (GPL) Version 3 * *****************************************************************************/ /* * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * for more details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ // **************** #include #include "osdgen.h" #include "attitudestate.h" #include "gpspositionsensor.h" #include "homelocation.h" #include "gpstime.h" #include "gpssatellites.h" #include "osdsettings.h" #include "barosensor.h" #include "taskinfo.h" #include "flightstatus.h" #include "fonts.h" #include "font12x18.h" #include "font8x10.h" #include "WMMInternal.h" #include "splash.h" /* static uint16_t angleA=0; static int16_t angleB=90; static int16_t angleC=0; static int16_t sum=2; static int16_t m_pitch=0; static int16_t m_roll=0; static int16_t m_yaw=0; static int16_t m_batt=0; static int16_t m_alt=0; static uint8_t m_gpsStatus=0; static int32_t m_gpsLat=0; static int32_t m_gpsLon=0; static float m_gpsAlt=0; static float m_gpsSpd=0;*/ extern uint8_t *draw_buffer_level; extern uint8_t *draw_buffer_mask; extern uint8_t *disp_buffer_level; extern uint8_t *disp_buffer_mask; TTime timex; // **************** // Private functions static void osdgenTask(void *parameters); // **************** // Private constants #define LONG_TIME 0xffff xSemaphoreHandle osdSemaphore = NULL; #define STACK_SIZE_BYTES 4096 #define TASK_PRIORITY (tskIDLE_PRIORITY + 4) #define UPDATE_PERIOD 100 // **************** // Private variables static xTaskHandle osdgenTaskHandle; struct splashEntry { unsigned int width, height; const uint16_t *level; const uint16_t *mask; }; struct splashEntry splash[3] = { { oplogo_width, oplogo_height, oplogo_bits, oplogo_mask_bits }, { level_width, level_height, level_bits, level_mask_bits }, { llama_width, llama_height, llama_bits, llama_mask_bits } }; uint16_t mirror(uint16_t source) { int result = ((source & 0x8000) >> 7) | ((source & 0x4000) >> 5) | ((source & 0x2000) >> 3) | ((source & 0x1000) >> 1) | ((source & 0x0800) << 1) | ((source & 0x0400) << 3) | ((source & 0x0200) << 5) | ((source & 0x0100) << 7) | ((source & 0x0080) >> 7) | ((source & 0x0040) >> 5) | ((source & 0x0020) >> 3) | ((source & 0x0010) >> 1) | ((source & 0x0008) << 1) | ((source & 0x0004) << 3) | ((source & 0x0002) << 5) | ((source & 0x0001) << 7); return result; } void clearGraphics() { memset((uint8_t *)draw_buffer_mask, 0, GRAPHICS_WIDTH * GRAPHICS_HEIGHT); memset((uint8_t *)draw_buffer_level, 0, GRAPHICS_WIDTH * GRAPHICS_HEIGHT); } void copyimage(uint16_t offsetx, uint16_t offsety, int image) { // check top/left position if (!validPos(offsetx, offsety)) { return; } struct splashEntry splash_info; splash_info = splash[image]; offsetx = offsetx / 8; for (uint16_t y = offsety; y < ((splash_info.height) + offsety); y++) { uint16_t x1 = offsetx; for (uint16_t x = offsetx; x < (((splash_info.width) / 16) + offsetx); x++) { draw_buffer_level[y * GRAPHICS_WIDTH + x1 + 1] = (uint8_t)( mirror(splash_info.level[(y - offsety) * ((splash_info.width) / 16) + (x - offsetx)]) >> 8); draw_buffer_level[y * GRAPHICS_WIDTH + x1] = (uint8_t)( mirror(splash_info.level[(y - offsety) * ((splash_info.width) / 16) + (x - offsetx)]) & 0xFF); draw_buffer_mask[y * GRAPHICS_WIDTH + x1 + 1] = (uint8_t)( mirror(splash_info.mask[(y - offsety) * ((splash_info.width) / 16) + (x - offsetx)]) >> 8); draw_buffer_mask[y * GRAPHICS_WIDTH + x1] = (uint8_t)(mirror(splash_info.mask[(y - offsety) * ((splash_info.width) / 16) + (x - offsetx)]) & 0xFF); x1 += 2; } } } uint8_t validPos(uint16_t x, uint16_t y) { if (x < GRAPHICS_HDEADBAND || x >= GRAPHICS_WIDTH_REAL || y >= GRAPHICS_HEIGHT_REAL) { return 0; } return 1; } // Credit for this one goes to wikipedia! :-) void drawCircle(uint16_t x0, uint16_t y0, uint16_t radius) { int f = 1 - radius; int ddF_x = 1; int ddF_y = -2 * radius; int x = 0; int y = radius; write_pixel_lm(x0, y0 + radius, 1, 1); write_pixel_lm(x0, y0 - radius, 1, 1); write_pixel_lm(x0 + radius, y0, 1, 1); write_pixel_lm(x0 - radius, y0, 1, 1); while (x < y) { // ddF_x == 2 * x + 1; // ddF_y == -2 * y; // f == x*x + y*y - radius*radius + 2*x - y + 1; if (f >= 0) { y--; ddF_y += 2; f += ddF_y; } x++; ddF_x += 2; f += ddF_x; write_pixel_lm(x0 + x, y0 + y, 1, 1); write_pixel_lm(x0 - x, y0 + y, 1, 1); write_pixel_lm(x0 + x, y0 - y, 1, 1); write_pixel_lm(x0 - x, y0 - y, 1, 1); write_pixel_lm(x0 + y, y0 + x, 1, 1); write_pixel_lm(x0 - y, y0 + x, 1, 1); write_pixel_lm(x0 + y, y0 - x, 1, 1); write_pixel_lm(x0 - y, y0 - x, 1, 1); } } void swap(uint16_t *a, uint16_t *b) { uint16_t temp = *a; *a = *b; *b = temp; } static const int8_t sinData[91] = { 0, 2, 3, 5, 7, 9, 10, 12, 14, 16, 17, 19, 21, 22, 24, 26, 28, 29, 31, 33, 34, 36, 37, 39, 41, 42, 44, 45, 47, 48, 50, 52, 53, 54, 56, 57, 59, 60, 62, 63, 64, 66, 67, 68, 69, 71, 72, 73, 74, 75, 77, 78, 79, 80, 81, 82, 83, 84, 85, 86, 87, 87, 88, 89, 90, 91, 91, 92, 93, 93, 94, 95, 95, 96, 96, 97, 97, 97, 98, 98, 98, 99, 99, 99, 99, 100, 100, 100, 100, 100, 100 }; static int8_t mySin(uint16_t angle) { uint16_t pos = 0; pos = angle % 360; int8_t mult = 1; // 180-359 is same as 0-179 but negative. if (pos >= 180) { pos = pos - 180; mult = -1; } // 0-89 is equal to 90-179 except backwards. if (pos >= 90) { pos = 180 - pos; } return mult * (int8_t)(sinData[pos]); } static int8_t myCos(uint16_t angle) { return mySin(angle + 90); } /// Draws four points relative to the given center point. /// /// \li centerX + X, centerY + Y /// \li centerX + X, centerY - Y /// \li centerX - X, centerY + Y /// \li centerX - X, centerY - Y /// /// \param centerX the x coordinate of the center point /// \param centerY the y coordinate of the center point /// \param deltaX the difference between the centerX coordinate and each pixel drawn /// \param deltaY the difference between the centerY coordinate and each pixel drawn /// \param color the color to draw the pixels with. void plotFourQuadrants(int32_t centerX, int32_t centerY, int32_t deltaX, int32_t deltaY) { write_pixel_lm(centerX + deltaX, centerY + deltaY, 1, 1); // Ist Quadrant write_pixel_lm(centerX - deltaX, centerY + deltaY, 1, 1); // IInd Quadrant write_pixel_lm(centerX - deltaX, centerY - deltaY, 1, 1); // IIIrd Quadrant write_pixel_lm(centerX + deltaX, centerY - deltaY, 1, 1); // IVth Quadrant } /// Implements the midpoint ellipse drawing algorithm which is a bresenham /// style DDF. /// /// \param centerX the x coordinate of the center of the ellipse /// \param centerY the y coordinate of the center of the ellipse /// \param horizontalRadius the horizontal radius of the ellipse /// \param verticalRadius the vertical radius of the ellipse /// \param color the color of the ellipse border void ellipse(int centerX, int centerY, int horizontalRadius, int verticalRadius) { int64_t doubleHorizontalRadius = horizontalRadius * horizontalRadius; int64_t doubleVerticalRadius = verticalRadius * verticalRadius; int64_t error = doubleVerticalRadius - doubleHorizontalRadius * verticalRadius + (doubleVerticalRadius >> 2); int x = 0; int y = verticalRadius; int deltaX = 0; int deltaY = (doubleHorizontalRadius << 1) * y; plotFourQuadrants(centerX, centerY, x, y); while (deltaY >= deltaX) { x++; deltaX += (doubleVerticalRadius << 1); error += deltaX + doubleVerticalRadius; if (error >= 0) { y--; deltaY -= (doubleHorizontalRadius << 1); error -= deltaY; } plotFourQuadrants(centerX, centerY, x, y); } error = (int64_t)(doubleVerticalRadius * (x + 1 / 2.0f) * (x + 1 / 2.0f) + doubleHorizontalRadius * (y - 1) * (y - 1) - doubleHorizontalRadius * doubleVerticalRadius); while (y >= 0) { error += doubleHorizontalRadius; y--; deltaY -= (doubleHorizontalRadius << 1); error -= deltaY; if (error <= 0) { x++; deltaX += (doubleVerticalRadius << 1); error += deltaX; } plotFourQuadrants(centerX, centerY, x, y); } } void drawArrow(uint16_t x, uint16_t y, uint16_t angle, uint16_t size) { int16_t a = myCos(angle); int16_t b = mySin(angle); a = (a * (size / 2)) / 100; b = (b * (size / 2)) / 100; write_line_lm((x) - 1 - b, (y) - 1 + a, (x) - 1 + b, (y) - 1 - a, 1, 1); // Direction line // write_line_lm((GRAPHICS_SIZE/2)-1 + a/2, (GRAPHICS_SIZE/2)-1 + b/2, (GRAPHICS_SIZE/2)-1 - a/2, (GRAPHICS_SIZE/2)-1 - b/2, 1, 1); //Arrow bottom line write_line_lm((x) - 1 + b, (y) - 1 - a, (x) - 1 - a / 2, (y) - 1 - b / 2, 1, 1); // Arrow "wings" write_line_lm((x) - 1 + b, (y) - 1 - a, (x) - 1 + a / 2, (y) - 1 + b / 2, 1, 1); } void drawBox(uint16_t x1, uint16_t y1, uint16_t x2, uint16_t y2) { write_line_lm(x1, y1, x2, y1, 1, 1); // top write_line_lm(x1, y1, x1, y2, 1, 1); // left write_line_lm(x2, y1, x2, y2, 1, 1); // right write_line_lm(x1, y2, x2, y2, 1, 1); // bottom } // simple routines // SUPEROSD routines, modified /** * write_pixel: Write a pixel at an x,y position to a given surface. * * @param buff pointer to buffer to write in * @param x x coordinate * @param y y coordinate * @param mode 0 = clear bit, 1 = set bit, 2 = toggle bit */ void write_pixel(uint8_t *buff, unsigned int x, unsigned int y, int mode) { CHECK_COORDS(x, y); // Determine the bit in the word to be set and the word // index to set it in. int bitnum = CALC_BIT_IN_WORD(x); int wordnum = CALC_BUFF_ADDR(x, y); // Apply a mask. uint16_t mask = 1 << (7 - bitnum); WRITE_WORD_MODE(buff, wordnum, mask, mode); } /** * write_pixel_lm: write the pixel on both surfaces (level and mask.) * Uses current draw buffer. * * @param x x coordinate * @param y y coordinate * @param mmode 0 = clear, 1 = set, 2 = toggle * @param lmode 0 = black, 1 = white, 2 = toggle */ void write_pixel_lm(unsigned int x, unsigned int y, int mmode, int lmode) { CHECK_COORDS(x, y); // Determine the bit in the word to be set and the word // index to set it in. int bitnum = CALC_BIT_IN_WORD(x); int wordnum = CALC_BUFF_ADDR(x, y); // Apply the masks. uint16_t mask = 1 << (7 - bitnum); WRITE_WORD_MODE(draw_buffer_mask, wordnum, mask, mmode); WRITE_WORD_MODE(draw_buffer_level, wordnum, mask, lmode); } /** * write_hline: optimised horizontal line writing algorithm * * @param buff pointer to buffer to write in * @param x0 x0 coordinate * @param x1 x1 coordinate * @param y y coordinate * @param mode 0 = clear, 1 = set, 2 = toggle */ void write_hline(uint8_t *buff, unsigned int x0, unsigned int x1, unsigned int y, int mode) { CLIP_COORDS(x0, y); CLIP_COORDS(x1, y); if (x0 > x1) { SWAP(x0, x1); } if (x0 == x1) { return; } /* This is an optimised algorithm for writing horizontal lines. * We begin by finding the addresses of the x0 and x1 points. */ int addr0 = CALC_BUFF_ADDR(x0, y); int addr1 = CALC_BUFF_ADDR(x1, y); int addr0_bit = CALC_BIT_IN_WORD(x0); int addr1_bit = CALC_BIT_IN_WORD(x1); int mask, mask_l, mask_r, i; /* If the addresses are equal, we only need to write one word * which is an island. */ if (addr0 == addr1) { mask = COMPUTE_HLINE_ISLAND_MASK(addr0_bit, addr1_bit); WRITE_WORD_MODE(buff, addr0, mask, mode); } else { /* Otherwise we need to write the edges and then the middle. */ mask_l = COMPUTE_HLINE_EDGE_L_MASK(addr0_bit); mask_r = COMPUTE_HLINE_EDGE_R_MASK(addr1_bit); WRITE_WORD_MODE(buff, addr0, mask_l, mode); WRITE_WORD_MODE(buff, addr1, mask_r, mode); // Now write 0xffff words from start+1 to end-1. for (i = addr0 + 1; i <= addr1 - 1; i++) { uint8_t m = 0xff; WRITE_WORD_MODE(buff, i, m, mode); } } } /** * write_hline_lm: write both level and mask buffers. * * @param x0 x0 coordinate * @param x1 x1 coordinate * @param y y coordinate * @param lmode 0 = clear, 1 = set, 2 = toggle * @param mmode 0 = clear, 1 = set, 2 = toggle */ void write_hline_lm(unsigned int x0, unsigned int x1, unsigned int y, int lmode, int mmode) { // TODO: an optimisation would compute the masks and apply to // both buffers simultaneously. write_hline(draw_buffer_level, x0, x1, y, lmode); write_hline(draw_buffer_mask, x0, x1, y, mmode); } /** * write_hline_outlined: outlined horizontal line with varying endcaps * Always uses draw buffer. * * @param x0 x0 coordinate * @param x1 x1 coordinate * @param y y coordinate * @param endcap0 0 = none, 1 = single pixel, 2 = full cap * @param endcap1 0 = none, 1 = single pixel, 2 = full cap * @param mode 0 = black outline, white body, 1 = white outline, black body * @param mmode 0 = clear, 1 = set, 2 = toggle */ void write_hline_outlined(unsigned int x0, unsigned int x1, unsigned int y, int endcap0, int endcap1, int mode, int mmode) { int stroke, fill; SETUP_STROKE_FILL(stroke, fill, mode) if (x0 > x1) { SWAP(x0, x1); } // Draw the main body of the line. write_hline_lm(x0 + 1, x1 - 1, y - 1, stroke, mmode); write_hline_lm(x0 + 1, x1 - 1, y + 1, stroke, mmode); write_hline_lm(x0 + 1, x1 - 1, y, fill, mmode); // Draw the endcaps, if any. DRAW_ENDCAP_HLINE(endcap0, x0, y, stroke, fill, mmode); DRAW_ENDCAP_HLINE(endcap1, x1, y, stroke, fill, mmode); } /** * write_vline: optimised vertical line writing algorithm * * @param buff pointer to buffer to write in * @param x x coordinate * @param y0 y0 coordinate * @param y1 y1 coordinate * @param mode 0 = clear, 1 = set, 2 = toggle */ void write_vline(uint8_t *buff, unsigned int x, unsigned int y0, unsigned int y1, int mode) { unsigned int a; CLIP_COORDS(x, y0); CLIP_COORDS(x, y1); if (y0 > y1) { SWAP(y0, y1); } if (y0 == y1) { return; } /* This is an optimised algorithm for writing vertical lines. * We begin by finding the addresses of the x,y0 and x,y1 points. */ unsigned int addr0 = CALC_BUFF_ADDR(x, y0); unsigned int addr1 = CALC_BUFF_ADDR(x, y1); /* Then we calculate the pixel data to be written. */ unsigned int bitnum = CALC_BIT_IN_WORD(x); uint16_t mask = 1 << (7 - bitnum); /* Run from addr0 to addr1 placing pixels. Increment by the number * of words n each graphics line. */ for (a = addr0; a <= addr1; a += GRAPHICS_WIDTH_REAL / 8) { WRITE_WORD_MODE(buff, a, mask, mode); } } /** * write_vline_lm: write both level and mask buffers. * * @param x x coordinate * @param y0 y0 coordinate * @param y1 y1 coordinate * @param lmode 0 = clear, 1 = set, 2 = toggle * @param mmode 0 = clear, 1 = set, 2 = toggle */ void write_vline_lm(unsigned int x, unsigned int y0, unsigned int y1, int lmode, int mmode) { // TODO: an optimisation would compute the masks and apply to // both buffers simultaneously. write_vline(draw_buffer_level, x, y0, y1, lmode); write_vline(draw_buffer_mask, x, y0, y1, mmode); } /** * write_vline_outlined: outlined vertical line with varying endcaps * Always uses draw buffer. * * @param x x coordinate * @param y0 y0 coordinate * @param y1 y1 coordinate * @param endcap0 0 = none, 1 = single pixel, 2 = full cap * @param endcap1 0 = none, 1 = single pixel, 2 = full cap * @param mode 0 = black outline, white body, 1 = white outline, black body * @param mmode 0 = clear, 1 = set, 2 = toggle */ void write_vline_outlined(unsigned int x, unsigned int y0, unsigned int y1, int endcap0, int endcap1, int mode, int mmode) { int stroke, fill; if (y0 > y1) { SWAP(y0, y1); } SETUP_STROKE_FILL(stroke, fill, mode); // Draw the main body of the line. write_vline_lm(x - 1, y0 + 1, y1 - 1, stroke, mmode); write_vline_lm(x + 1, y0 + 1, y1 - 1, stroke, mmode); write_vline_lm(x, y0 + 1, y1 - 1, fill, mmode); // Draw the endcaps, if any. DRAW_ENDCAP_VLINE(endcap0, x, y0, stroke, fill, mmode); DRAW_ENDCAP_VLINE(endcap1, x, y1, stroke, fill, mmode); } /** * write_filled_rectangle: draw a filled rectangle. * * Uses an optimised algorithm which is similar to the horizontal * line writing algorithm, but optimised for writing the lines * multiple times without recalculating lots of stuff. * * @param buff pointer to buffer to write in * @param x x coordinate (left) * @param y y coordinate (top) * @param width rectangle width * @param height rectangle height * @param mode 0 = clear, 1 = set, 2 = toggle */ void write_filled_rectangle(uint8_t *buff, unsigned int x, unsigned int y, unsigned int width, unsigned int height, int mode) { unsigned int yy, addr0_old, addr1_old; CHECK_COORDS(x, y); CHECK_COORD_X(x + width); CHECK_COORD_Y(y + height); if (width <= 0 || height <= 0) { return; } // Calculate as if the rectangle was only a horizontal line. We then // step these addresses through each row until we iterate `height` times. unsigned int addr0 = CALC_BUFF_ADDR(x, y); unsigned int addr1 = CALC_BUFF_ADDR(x + width, y); unsigned int addr0_bit = CALC_BIT_IN_WORD(x); unsigned int addr1_bit = CALC_BIT_IN_WORD(x + width); unsigned int mask, mask_l, mask_r, i; // If the addresses are equal, we need to write one word vertically. if (addr0 == addr1) { mask = COMPUTE_HLINE_ISLAND_MASK(addr0_bit, addr1_bit); while (height--) { WRITE_WORD_MODE(buff, addr0, mask, mode); addr0 += GRAPHICS_WIDTH_REAL / 8; } } else { // Otherwise we need to write the edges and then the middle repeatedly. mask_l = COMPUTE_HLINE_EDGE_L_MASK(addr0_bit); mask_r = COMPUTE_HLINE_EDGE_R_MASK(addr1_bit); // Write edges first. yy = 0; addr0_old = addr0; addr1_old = addr1; while (yy < height) { WRITE_WORD_MODE(buff, addr0, mask_l, mode); WRITE_WORD_MODE(buff, addr1, mask_r, mode); addr0 += GRAPHICS_WIDTH_REAL / 8; addr1 += GRAPHICS_WIDTH_REAL / 8; yy++; } // Now write 0xffff words from start+1 to end-1 for each row. yy = 0; addr0 = addr0_old; addr1 = addr1_old; while (yy < height) { for (i = addr0 + 1; i <= addr1 - 1; i++) { uint8_t m = 0xff; WRITE_WORD_MODE(buff, i, m, mode); } addr0 += GRAPHICS_WIDTH_REAL / 8; addr1 += GRAPHICS_WIDTH_REAL / 8; yy++; } } } /** * write_filled_rectangle_lm: draw a filled rectangle on both draw buffers. * * @param x x coordinate (left) * @param y y coordinate (top) * @param width rectangle width * @param height rectangle height * @param lmode 0 = clear, 1 = set, 2 = toggle * @param mmode 0 = clear, 1 = set, 2 = toggle */ void write_filled_rectangle_lm(unsigned int x, unsigned int y, unsigned int width, unsigned int height, int lmode, int mmode) { write_filled_rectangle(draw_buffer_mask, x, y, width, height, mmode); write_filled_rectangle(draw_buffer_level, x, y, width, height, lmode); } /** * write_rectangle_outlined: draw an outline of a rectangle. Essentially * a convenience wrapper for draw_hline_outlined and draw_vline_outlined. * * @param x x coordinate (left) * @param y y coordinate (top) * @param width rectangle width * @param height rectangle height * @param mode 0 = black outline, white body, 1 = white outline, black body * @param mmode 0 = clear, 1 = set, 2 = toggle */ void write_rectangle_outlined(unsigned int x, unsigned int y, int width, int height, int mode, int mmode) { // CHECK_COORDS(x, y); // CHECK_COORDS(x + width, y + height); // if((x + width) > DISP_WIDTH) width = DISP_WIDTH - x; // if((y + height) > DISP_HEIGHT) height = DISP_HEIGHT - y; write_hline_outlined(x, x + width, y, ENDCAP_ROUND, ENDCAP_ROUND, mode, mmode); write_hline_outlined(x, x + width, y + height, ENDCAP_ROUND, ENDCAP_ROUND, mode, mmode); write_vline_outlined(x, y, y + height, ENDCAP_ROUND, ENDCAP_ROUND, mode, mmode); write_vline_outlined(x + width, y, y + height, ENDCAP_ROUND, ENDCAP_ROUND, mode, mmode); } /** * write_circle: draw the outline of a circle on a given buffer, * with an optional dash pattern for the line instead of a normal line. * * @param buff pointer to buffer to write in * @param cx origin x coordinate * @param cy origin y coordinate * @param r radius * @param dashp dash period (pixels) - zero for no dash * @param mode 0 = clear, 1 = set, 2 = toggle */ void write_circle(uint8_t *buff, unsigned int cx, unsigned int cy, unsigned int r, unsigned int dashp, int mode) { CHECK_COORDS(cx, cy); int error = -r, x = r, y = 0; while (x >= y) { if (dashp == 0 || (y % dashp) < (dashp / 2)) { CIRCLE_PLOT_8(buff, cx, cy, x, y, mode); } error += (y * 2) + 1; y++; if (error >= 0) { --x; error -= x * 2; } } } /** * write_circle_outlined: draw an outlined circle on the draw buffer. * * @param cx origin x coordinate * @param cy origin y coordinate * @param r radius * @param dashp dash period (pixels) - zero for no dash * @param bmode 0 = 4-neighbour border, 1 = 8-neighbour border * @param mode 0 = black outline, white body, 1 = white outline, black body * @param mmode 0 = clear, 1 = set, 2 = toggle */ void write_circle_outlined(unsigned int cx, unsigned int cy, unsigned int r, unsigned int dashp, int bmode, int mode, int mmode) { int stroke, fill; CHECK_COORDS(cx, cy); SETUP_STROKE_FILL(stroke, fill, mode); // This is a two step procedure. First, we draw the outline of the // circle, then we draw the inner part. int error = -r, x = r, y = 0; while (x >= y) { if (dashp == 0 || (y % dashp) < (dashp / 2)) { CIRCLE_PLOT_8(draw_buffer_mask, cx, cy, x + 1, y, mmode); CIRCLE_PLOT_8(draw_buffer_level, cx, cy, x + 1, y, stroke); CIRCLE_PLOT_8(draw_buffer_mask, cx, cy, x, y + 1, mmode); CIRCLE_PLOT_8(draw_buffer_level, cx, cy, x, y + 1, stroke); CIRCLE_PLOT_8(draw_buffer_mask, cx, cy, x - 1, y, mmode); CIRCLE_PLOT_8(draw_buffer_level, cx, cy, x - 1, y, stroke); CIRCLE_PLOT_8(draw_buffer_mask, cx, cy, x, y - 1, mmode); CIRCLE_PLOT_8(draw_buffer_level, cx, cy, x, y - 1, stroke); if (bmode == 1) { CIRCLE_PLOT_8(draw_buffer_mask, cx, cy, x + 1, y + 1, mmode); CIRCLE_PLOT_8(draw_buffer_level, cx, cy, x + 1, y + 1, stroke); CIRCLE_PLOT_8(draw_buffer_mask, cx, cy, x - 1, y - 1, mmode); CIRCLE_PLOT_8(draw_buffer_level, cx, cy, x - 1, y - 1, stroke); } } error += (y * 2) + 1; y++; if (error >= 0) { --x; error -= x * 2; } } error = -r; x = r; y = 0; while (x >= y) { if (dashp == 0 || (y % dashp) < (dashp / 2)) { CIRCLE_PLOT_8(draw_buffer_mask, cx, cy, x, y, mmode); CIRCLE_PLOT_8(draw_buffer_level, cx, cy, x, y, fill); } error += (y * 2) + 1; y++; if (error >= 0) { --x; error -= x * 2; } } } /** * write_circle_filled: fill a circle on a given buffer. * * @param buff pointer to buffer to write in * @param cx origin x coordinate * @param cy origin y coordinate * @param r radius * @param mode 0 = clear, 1 = set, 2 = toggle */ void write_circle_filled(uint8_t *buff, unsigned int cx, unsigned int cy, unsigned int r, int mode) { CHECK_COORDS(cx, cy); int error = -r, x = r, y = 0, xch = 0; // It turns out that filled circles can take advantage of the midpoint // circle algorithm. We simply draw very fast horizontal lines across each // pair of X,Y coordinates. In some cases, this can even be faster than // drawing an outlined circle! // // Due to multiple writes to each set of pixels, we have a special exception // for when using the toggling draw mode. while (x >= y) { if (y != 0) { write_hline(buff, cx - x, cx + x, cy + y, mode); write_hline(buff, cx - x, cx + x, cy - y, mode); if (mode != 2 || (mode == 2 && xch && (cx - x) != (cx - y))) { write_hline(buff, cx - y, cx + y, cy + x, mode); write_hline(buff, cx - y, cx + y, cy - x, mode); xch = 0; } } error += (y * 2) + 1; y++; if (error >= 0) { --x; xch = 1; error -= x * 2; } } // Handle toggle mode. if (mode == 2) { write_hline(buff, cx - r, cx + r, cy, mode); } } /** * write_line: Draw a line of arbitrary angle. * * @param buff pointer to buffer to write in * @param x0 first x coordinate * @param y0 first y coordinate * @param x1 second x coordinate * @param y1 second y coordinate * @param mode 0 = clear, 1 = set, 2 = toggle */ void write_line(uint8_t *buff, unsigned int x0, unsigned int y0, unsigned int x1, unsigned int y1, int mode) { // Based on http://en.wikipedia.org/wiki/Bresenham%27s_line_algorithm unsigned int steep = abs(y1 - y0) > abs(x1 - x0); if (steep) { SWAP(x0, y0); SWAP(x1, y1); } if (x0 > x1) { SWAP(x0, x1); SWAP(y0, y1); } int deltax = x1 - x0; unsigned int deltay = abs(y1 - y0); int error = deltax / 2; int ystep; unsigned int y = y0; unsigned int x; // , lasty = y, stox = 0; if (y0 < y1) { ystep = 1; } else { ystep = -1; } for (x = x0; x < x1; x++) { if (steep) { write_pixel(buff, y, x, mode); } else { write_pixel(buff, x, y, mode); } error -= deltay; if (error < 0) { y += ystep; error += deltax; } } } /** * write_line_lm: Draw a line of arbitrary angle. * * @param x0 first x coordinate * @param y0 first y coordinate * @param x1 second x coordinate * @param y1 second y coordinate * @param mmode 0 = clear, 1 = set, 2 = toggle * @param lmode 0 = clear, 1 = set, 2 = toggle */ void write_line_lm(unsigned int x0, unsigned int y0, unsigned int x1, unsigned int y1, int mmode, int lmode) { write_line(draw_buffer_mask, x0, y0, x1, y1, mmode); write_line(draw_buffer_level, x0, y0, x1, y1, lmode); } /** * write_line_outlined: Draw a line of arbitrary angle, with an outline. * * @param buff pointer to buffer to write in * @param x0 first x coordinate * @param y0 first y coordinate * @param x1 second x coordinate * @param y1 second y coordinate * @param endcap0 0 = none, 1 = single pixel, 2 = full cap * @param endcap1 0 = none, 1 = single pixel, 2 = full cap * @param mode 0 = black outline, white body, 1 = white outline, black body * @param mmode 0 = clear, 1 = set, 2 = toggle */ void write_line_outlined(unsigned int x0, unsigned int y0, unsigned int x1, unsigned int y1, __attribute__((unused)) int endcap0, __attribute__((unused)) int endcap1, int mode, int mmode) { // Based on http://en.wikipedia.org/wiki/Bresenham%27s_line_algorithm // This could be improved for speed. int omode, imode; if (mode == 0) { omode = 0; imode = 1; } else { omode = 1; imode = 0; } int steep = abs(y1 - y0) > abs(x1 - x0); if (steep) { SWAP(x0, y0); SWAP(x1, y1); } if (x0 > x1) { SWAP(x0, x1); SWAP(y0, y1); } int deltax = x1 - x0; unsigned int deltay = abs(y1 - y0); int error = deltax / 2; int ystep; unsigned int y = y0; unsigned int x; if (y0 < y1) { ystep = 1; } else { ystep = -1; } // Draw the outline. for (x = x0; x < x1; x++) { if (steep) { write_pixel_lm(y - 1, x, mmode, omode); write_pixel_lm(y + 1, x, mmode, omode); write_pixel_lm(y, x - 1, mmode, omode); write_pixel_lm(y, x + 1, mmode, omode); } else { write_pixel_lm(x - 1, y, mmode, omode); write_pixel_lm(x + 1, y, mmode, omode); write_pixel_lm(x, y - 1, mmode, omode); write_pixel_lm(x, y + 1, mmode, omode); } error -= deltay; if (error < 0) { y += ystep; error += deltax; } } // Now draw the innards. error = deltax / 2; y = y0; for (x = x0; x < x1; x++) { if (steep) { write_pixel_lm(y, x, mmode, imode); } else { write_pixel_lm(x, y, mmode, imode); } error -= deltay; if (error < 0) { y += ystep; error += deltax; } } } /** * write_word_misaligned: Write a misaligned word across two addresses * with an x offset. * * This allows for many pixels to be set in one write. * * @param buff buffer to write in * @param word word to write (16 bits) * @param addr address of first word * @param xoff x offset (0-15) * @param mode 0 = clear, 1 = set, 2 = toggle */ void write_word_misaligned(uint8_t *buff, uint16_t word, unsigned int addr, unsigned int xoff, int mode) { int16_t firstmask = word >> xoff; int16_t lastmask = word << (16 - xoff); WRITE_WORD_MODE(buff, addr + 1, firstmask && 0x00ff, mode); WRITE_WORD_MODE(buff, addr, (firstmask & 0xff00) >> 8, mode); if (xoff > 0) { WRITE_WORD_MODE(buff, addr + 2, (lastmask & 0xff00) >> 8, mode); } } /** * write_word_misaligned_NAND: Write a misaligned word across two addresses * with an x offset, using a NAND mask. * * This allows for many pixels to be set in one write. * * @param buff buffer to write in * @param word word to write (16 bits) * @param addr address of first word * @param xoff x offset (0-15) * * This is identical to calling write_word_misaligned with a mode of 0 but * it doesn't go through a lot of switch logic which slows down text writing * a lot. */ void write_word_misaligned_NAND(uint8_t *buff, uint16_t word, unsigned int addr, unsigned int xoff) { uint16_t firstmask = word >> xoff; uint16_t lastmask = word << (16 - xoff); WRITE_WORD_NAND(buff, addr + 1, firstmask & 0x00ff); WRITE_WORD_NAND(buff, addr, (firstmask & 0xff00) >> 8); if (xoff > 0) { WRITE_WORD_NAND(buff, addr + 2, (lastmask & 0xff00) >> 8); } } /** * write_word_misaligned_OR: Write a misaligned word across two addresses * with an x offset, using an OR mask. * * This allows for many pixels to be set in one write. * * @param buff buffer to write in * @param word word to write (16 bits) * @param addr address of first word * @param xoff x offset (0-15) * * This is identical to calling write_word_misaligned with a mode of 1 but * it doesn't go through a lot of switch logic which slows down text writing * a lot. */ void write_word_misaligned_OR(uint8_t *buff, uint16_t word, unsigned int addr, unsigned int xoff) { uint16_t firstmask = word >> xoff; uint16_t lastmask = word << (16 - xoff); WRITE_WORD_OR(buff, addr + 1, firstmask & 0x00ff); WRITE_WORD_OR(buff, addr, (firstmask & 0xff00) >> 8); if (xoff > 0) { WRITE_WORD_OR(buff, addr + 2, (lastmask & 0xff00) >> 8); } } /** * write_word_misaligned_lm: Write a misaligned word across two * words, in both level and mask buffers. This is core to the text * writing routines. * * @param buff buffer to write in * @param word word to write (16 bits) * @param addr address of first word * @param xoff x offset (0-15) * @param lmode 0 = clear, 1 = set, 2 = toggle * @param mmode 0 = clear, 1 = set, 2 = toggle */ void write_word_misaligned_lm(uint16_t wordl, uint16_t wordm, unsigned int addr, unsigned int xoff, int lmode, int mmode) { write_word_misaligned(draw_buffer_level, wordl, addr, xoff, lmode); write_word_misaligned(draw_buffer_mask, wordm, addr, xoff, mmode); } /** * fetch_font_info: Fetch font info structs. * * @param ch character * @param font font id */ int fetch_font_info(uint8_t ch, int font, struct FontEntry *font_info, char *lookup) { // First locate the font struct. if ((unsigned int)font > SIZEOF_ARRAY(fonts)) { return 0; // font does not exist, exit. } // Load the font info; IDs are always sequential. *font_info = fonts[font]; // Locate character in font lookup table. (If required.) if (lookup != NULL) { *lookup = font_info->lookup[ch]; if (*lookup == 0xff) { return 0; // character doesn't exist, don't bother writing it. } } return 1; } /** * write_char16: Draw a character on the current draw buffer. * Currently supports outlined characters and characters with * a width of up to 8 pixels. * * @param ch character to write * @param x x coordinate (left) * @param y y coordinate (top) * @param flags flags to write with (see gfx.h) * @param font font to use */ void write_char16(char ch, unsigned int x, unsigned int y, int font) { unsigned int yy, addr_temp, row, row_temp, xshift; uint16_t and_mask, or_mask, levels; struct FontEntry font_info; // char lookup = 0; fetch_font_info(0, font, &font_info, NULL); // Compute starting address (for x,y) of character. int addr = CALC_BUFF_ADDR(x, y); int wbit = CALC_BIT_IN_WORD(x); // If font only supports lowercase or uppercase, make the letter // lowercase or uppercase. // How big is the character? We handle characters up to 8 pixels // wide for now. Support for large characters may be added in future. { // Ensure we don't overflow. if (x + wbit > GRAPHICS_WIDTH_REAL) { return; } // Load data pointer. row = ch * font_info.height; row_temp = row; addr_temp = addr; xshift = 16 - font_info.width; // We can write mask words easily. for (yy = y; yy < y + font_info.height; yy++) { if (font == 3) { write_word_misaligned_OR(draw_buffer_mask, font_mask12x18[row] << xshift, addr, wbit); } else { write_word_misaligned_OR(draw_buffer_mask, font_mask8x10[row] << xshift, addr, wbit); } addr += GRAPHICS_WIDTH_REAL / 8; row++; } // Level bits are more complicated. We need to set or clear // level bits, but only where the mask bit is set; otherwise, // we need to leave them alone. To do this, for each word, we // construct an AND mask and an OR mask, and apply each individually. row = row_temp; addr = addr_temp; for (yy = y; yy < y + font_info.height; yy++) { if (font == 3) { levels = font_frame12x18[row]; // if(!(flags & FONT_INVERT)) // data is normally inverted levels = ~levels; or_mask = font_mask12x18[row] << xshift; and_mask = (font_mask12x18[row] & levels) << xshift; } else { levels = font_frame8x10[row]; // if(!(flags & FONT_INVERT)) // data is normally inverted levels = ~levels; or_mask = font_mask8x10[row] << xshift; and_mask = (font_mask8x10[row] & levels) << xshift; } write_word_misaligned_OR(draw_buffer_level, or_mask, addr, wbit); // If we're not bold write the AND mask. // if(!(flags & FONT_BOLD)) write_word_misaligned_NAND(draw_buffer_level, and_mask, addr, wbit); addr += GRAPHICS_WIDTH_REAL / 8; row++; } } } /** * write_char: Draw a character on the current draw buffer. * Currently supports outlined characters and characters with * a width of up to 8 pixels. * * @param ch character to write * @param x x coordinate (left) * @param y y coordinate (top) * @param flags flags to write with (see gfx.h) * @param font font to use */ void write_char(char ch, unsigned int x, unsigned int y, int flags, int font) { unsigned int yy, addr_temp, row, row_temp, xshift; uint16_t and_mask, or_mask, levels; struct FontEntry font_info; char lookup = 0; fetch_font_info(ch, font, &font_info, &lookup); // Compute starting address (for x,y) of character. unsigned int addr = CALC_BUFF_ADDR(x, y); unsigned int wbit = CALC_BIT_IN_WORD(x); // If font only supports lowercase or uppercase, make the letter // lowercase or uppercase. /*if(font_info.flags & FONT_LOWERCASE_ONLY) ch = tolower(ch); if(font_info.flags & FONT_UPPERCASE_ONLY) ch = toupper(ch);*/ fetch_font_info(ch, font, &font_info, &lookup); // How big is the character? We handle characters up to 8 pixels // wide for now. Support for large characters may be added in future. if (font_info.width <= 8) { // Ensure we don't overflow. if (x + wbit > GRAPHICS_WIDTH_REAL) { return; } // Load data pointer. row = lookup * font_info.height * 2; row_temp = row; addr_temp = addr; xshift = 16 - font_info.width; // We can write mask words easily. for (yy = y; yy < y + font_info.height; yy++) { write_word_misaligned_OR(draw_buffer_mask, font_info.data[row] << xshift, addr, wbit); addr += GRAPHICS_WIDTH_REAL / 8; row++; } // Level bits are more complicated. We need to set or clear // level bits, but only where the mask bit is set; otherwise, // we need to leave them alone. To do this, for each word, we // construct an AND mask and an OR mask, and apply each individually. row = row_temp; addr = addr_temp; for (yy = y; yy < y + font_info.height; yy++) { levels = font_info.data[row + font_info.height]; if (!(flags & FONT_INVERT)) { // data is normally inverted levels = ~levels; } or_mask = font_info.data[row] << xshift; and_mask = (font_info.data[row] & levels) << xshift; write_word_misaligned_OR(draw_buffer_level, or_mask, addr, wbit); // If we're not bold write the AND mask. // if(!(flags & FONT_BOLD)) write_word_misaligned_NAND(draw_buffer_level, and_mask, addr, wbit); addr += GRAPHICS_WIDTH_REAL / 8; row++; } } } /** * calc_text_dimensions: Calculate the dimensions of a * string in a given font. Supports new lines and * carriage returns in text. * * @param str string to calculate dimensions of * @param font_info font info structure * @param xs horizontal spacing * @param ys vertical spacing * @param dim return result: struct FontDimensions */ void calc_text_dimensions(char *str, struct FontEntry font, int xs, int ys, struct FontDimensions *dim) { int max_length = 0, line_length = 0, lines = 1; while (*str != 0) { line_length++; if (*str == '\n' || *str == '\r') { if (line_length > max_length) { max_length = line_length; } line_length = 0; lines++; } str++; } if (line_length > max_length) { max_length = line_length; } dim->width = max_length * (font.width + xs); dim->height = lines * (font.height + ys); } /** * write_string: Draw a string on the screen with certain * alignment parameters. * * @param str string to write * @param x x coordinate * @param y y coordinate * @param xs horizontal spacing * @param ys horizontal spacing * @param va vertical align * @param ha horizontal align * @param flags flags (passed to write_char) * @param font font */ void write_string(char *str, unsigned int x, unsigned int y, unsigned int xs, unsigned int ys, int va, int ha, int flags, int font) { int xx = 0, yy = 0, xx_original = 0; struct FontEntry font_info; struct FontDimensions dim; // Determine font info and dimensions/position of the string. fetch_font_info(0, font, &font_info, NULL); calc_text_dimensions(str, font_info, xs, ys, &dim); switch (va) { case TEXT_VA_TOP: yy = y; break; case TEXT_VA_MIDDLE: yy = y - (dim.height / 2); break; case TEXT_VA_BOTTOM: yy = y - dim.height; break; } switch (ha) { case TEXT_HA_LEFT: xx = x; break; case TEXT_HA_CENTER: xx = x - (dim.width / 2); break; case TEXT_HA_RIGHT: xx = x - dim.width; break; } // Then write each character. xx_original = xx; while (*str != 0) { if (*str == '\n' || *str == '\r') { yy += ys + font_info.height; xx = xx_original; } else { if (xx >= 0 && xx < GRAPHICS_WIDTH_REAL) { if (font_info.id < 2) { write_char(*str, xx, yy, flags, font); } else { write_char16(*str, xx, yy, font); } } xx += font_info.width + xs; } str++; } } /** * write_string_formatted: Draw a string with format escape * sequences in it. Allows for complex text effects. * * @param str string to write (with format data) * @param x x coordinate * @param y y coordinate * @param xs default horizontal spacing * @param ys default horizontal spacing * @param va vertical align * @param ha horizontal align * @param flags flags (passed to write_char) */ void write_string_formatted(char *str, unsigned int x, unsigned int y, unsigned int xs, unsigned int ys, __attribute__((unused)) int va, __attribute__((unused)) int ha, int flags) { int fcode = 0, fptr = 0, font = 0, fwidth = 0, fheight = 0, xx = x, yy = y, max_xx = 0, max_height = 0; struct FontEntry font_info; // Retrieve sizes of the fonts: bigfont and smallfont. fetch_font_info(0, 0, &font_info, NULL); int smallfontwidth = font_info.width, smallfontheight = font_info.height; fetch_font_info(0, 1, &font_info, NULL); int bigfontwidth = font_info.width, bigfontheight = font_info.height; // 11 byte stack with last byte as NUL. char fstack[11]; fstack[10] = '\0'; // First, we need to parse the string for format characters and // work out a bounding box. We'll parse again for the final output. // This is a simple state machine parser. char *ostr = str; while (*str) { if (*str == '<' && fcode == 1) { // escape code: skip fcode = 0; } if (*str == '<' && fcode == 0) { // begin format code? fcode = 1; fptr = 0; } if (*str == '>' && fcode == 1) { fcode = 0; if (strcmp(fstack, "B")) { // switch to "big" font (font #1) fwidth = bigfontwidth; fheight = bigfontheight; } else if (strcmp(fstack, "S")) { // switch to "small" font (font #0) fwidth = smallfontwidth; fheight = smallfontheight; } if (fheight > max_height) { max_height = fheight; } // Skip over this byte. Go to next byte. str++; continue; } if (*str != '<' && *str != '>' && fcode == 1) { // Add to the format stack (up to 10 bytes.) if (fptr > 10) { // stop adding bytes str++; // go to next byte continue; } fstack[fptr++] = *str; fstack[fptr] = '\0'; // clear next byte (ready for next char or to terminate string.) } if (fcode == 0) { // Not a format code, raw text. xx += fwidth + xs; if (*str == '\n') { if (xx > max_xx) { max_xx = xx; } xx = x; yy += fheight + ys; } } str++; } // Reset string pointer. str = ostr; // Now we've parsed it and got a bbox, we need to work out the dimensions of it // and how to align it. /*int width = max_xx - x; int height = yy - y; int ay, ax; switch(va) { case TEXT_VA_TOP: ay = yy; break; case TEXT_VA_MIDDLE: ay = yy - (height / 2); break; case TEXT_VA_BOTTOM: ay = yy - height; break; } switch(ha) { case TEXT_HA_LEFT: ax = x; break; case TEXT_HA_CENTER: ax = x - (width / 2); break; case TEXT_HA_RIGHT: ax = x - width; break; }*/ // So ax,ay is our new text origin. Parse the text format again and paint // the text on the display. fcode = 0; fptr = 0; font = 0; xx = 0; yy = 0; while (*str) { if (*str == '<' && fcode == 1) { // escape code: skip fcode = 0; } if (*str == '<' && fcode == 0) { // begin format code? fcode = 1; fptr = 0; } if (*str == '>' && fcode == 1) { fcode = 0; if (strcmp(fstack, "B")) { // switch to "big" font (font #1) fwidth = bigfontwidth; fheight = bigfontheight; font = 1; } else if (strcmp(fstack, "S")) { // switch to "small" font (font #0) fwidth = smallfontwidth; fheight = smallfontheight; font = 0; } // Skip over this byte. Go to next byte. str++; continue; } if (*str != '<' && *str != '>' && fcode == 1) { // Add to the format stack (up to 10 bytes.) if (fptr > 10) { // stop adding bytes str++; // go to next byte continue; } fstack[fptr++] = *str; fstack[fptr] = '\0'; // clear next byte (ready for next char or to terminate string.) } if (fcode == 0) { // Not a format code, raw text. So we draw it. // TODO - different font sizes. write_char(*str, xx, yy + (max_height - fheight), flags, font); xx += fwidth + xs; if (*str == '\n') { if (xx > max_xx) { max_xx = xx; } xx = x; yy += fheight + ys; } } str++; } } // SUPEROSD- // graphics void drawAttitude(uint16_t x, uint16_t y, int16_t pitch, int16_t roll, uint16_t size) { int16_t a = mySin(roll + 360); int16_t b = myCos(roll + 360); int16_t c = mySin(roll + 90 + 360) * 5 / 100; int16_t d = myCos(roll + 90 + 360) * 5 / 100; int16_t k; int16_t l; int16_t indi30x1 = myCos(30) * (size / 2 + 1) / 100; int16_t indi30y1 = mySin(30) * (size / 2 + 1) / 100; int16_t indi30x2 = myCos(30) * (size / 2 + 4) / 100; int16_t indi30y2 = mySin(30) * (size / 2 + 4) / 100; int16_t indi60x1 = myCos(60) * (size / 2 + 1) / 100; int16_t indi60y1 = mySin(60) * (size / 2 + 1) / 100; int16_t indi60x2 = myCos(60) * (size / 2 + 4) / 100; int16_t indi60y2 = mySin(60) * (size / 2 + 4) / 100; pitch = pitch % 90; if (pitch > 90) { pitch = pitch - 90; } if (pitch < -90) { pitch = pitch + 90; } a = (a * (size / 2)) / 100; b = (b * (size / 2)) / 100; if (roll < -90 || roll > 90) { pitch = pitch * -1; } k = a * pitch / 90; l = b * pitch / 90; // scale // 0 // drawLine((x)-1-(size/2+4), (y)-1, (x)-1 - (size/2+1), (y)-1); // drawLine((x)-1+(size/2+4), (y)-1, (x)-1 + (size/2+1), (y)-1); write_line_outlined((x) - 1 - (size / 2 + 4), (y) - 1, (x) - 1 - (size / 2 + 1), (y) - 1, 0, 0, 0, 1); write_line_outlined((x) - 1 + (size / 2 + 4), (y) - 1, (x) - 1 + (size / 2 + 1), (y) - 1, 0, 0, 0, 1); // 30 // drawLine((x)-1+indi30x1, (y)-1-indi30y1, (x)-1 + indi30x2, (y)-1 - indi30y2); // drawLine((x)-1-indi30x1, (y)-1-indi30y1, (x)-1 - indi30x2, (y)-1 - indi30y2); write_line_outlined((x) - 1 + indi30x1, (y) - 1 - indi30y1, (x) - 1 + indi30x2, (y) - 1 - indi30y2, 0, 0, 0, 1); write_line_outlined((x) - 1 - indi30x1, (y) - 1 - indi30y1, (x) - 1 - indi30x2, (y) - 1 - indi30y2, 0, 0, 0, 1); // 60 // drawLine((x)-1+indi60x1, (y)-1-indi60y1, (x)-1 + indi60x2, (y)-1 - indi60y2); // drawLine((x)-1-indi60x1, (y)-1-indi60y1, (x)-1 - indi60x2, (y)-1 - indi60y2); write_line_outlined((x) - 1 + indi60x1, (y) - 1 - indi60y1, (x) - 1 + indi60x2, (y) - 1 - indi60y2, 0, 0, 0, 1); write_line_outlined((x) - 1 - indi60x1, (y) - 1 - indi60y1, (x) - 1 - indi60x2, (y) - 1 - indi60y2, 0, 0, 0, 1); // 90 // drawLine((x)-1, (y)-1-(size/2+4), (x)-1, (y)-1 - (size/2+1)); write_line_outlined((x) - 1, (y) - 1 - (size / 2 + 4), (x) - 1, (y) - 1 - (size / 2 + 1), 0, 0, 0, 1); // roll // drawLine((x)-1 - b, (y)-1 + a, (x)-1 + b, (y)-1 - a); //Direction line write_line_outlined((x) - 1 - b, (y) - 1 + a, (x) - 1 + b, (y) - 1 - a, 0, 0, 0, 1); // Direction line // "wingtips" // drawLine((x)-1 - b, (y)-1 + a, (x)-1 - b + d, (y)-1 + a - c); // drawLine((x)-1 + b + d, (y)-1 - a - c, (x)-1 + b, (y)-1 - a); write_line_outlined((x) - 1 - b, (y) - 1 + a, (x) - 1 - b + d, (y) - 1 + a - c, 0, 0, 0, 1); write_line_outlined((x) - 1 + b + d, (y) - 1 - a - c, (x) - 1 + b, (y) - 1 - a, 0, 0, 0, 1); // pitch // drawLine((x)-1, (y)-1, (x)-1 - k, (y)-1 - l); write_line_outlined((x) - 1, (y) - 1, (x) - 1 - k, (y) - 1 - l, 0, 0, 0, 1); // drawCircle(x-1, y-1, 5); // write_circle_outlined(x-1, y-1, 5,0,0,0,1); // drawCircle(x-1, y-1, size/2+4); // write_circle_outlined(x-1, y-1, size/2+4,0,0,0,1); } void drawBattery(uint16_t x, uint16_t y, uint8_t battery, uint16_t size) { int i = 0; int batteryLines; // top /*drawLine((x)-1+(size/2-size/4), (y)-1, (x)-1 + (size/2+size/4), (y)-1); drawLine((x)-1+(size/2-size/4), (y)-1+1, (x)-1 + (size/2+size/4), (y)-1+1); drawLine((x)-1, (y)-1+2, (x)-1 + size, (y)-1+2); //bottom drawLine((x)-1, (y)-1+size*3, (x)-1 + size, (y)-1+size*3); //left drawLine((x)-1, (y)-1+2, (x)-1, (y)-1+size*3); //right drawLine((x)-1+size, (y)-1+2, (x)-1+size, (y)-1+size*3);*/ write_rectangle_outlined((x) - 1, (y) - 1 + 2, size, size * 3, 0, 1); write_vline_lm((x) - 1 + (size / 2 + size / 4) + 1, (y) - 2, (y) - 1 + 1, 0, 1); write_vline_lm((x) - 1 + (size / 2 - size / 4) - 1, (y) - 2, (y) - 1 + 1, 0, 1); write_hline_lm((x) - 1 + (size / 2 - size / 4), (x) - 1 + (size / 2 + size / 4), (y) - 2, 0, 1); write_hline_lm((x) - 1 + (size / 2 - size / 4), (x) - 1 + (size / 2 + size / 4), (y) - 1, 1, 1); write_hline_lm((x) - 1 + (size / 2 - size / 4), (x) - 1 + (size / 2 + size / 4), (y) - 1 + 1, 1, 1); batteryLines = battery * (size * 3 - 2) / 100; for (i = 0; i < batteryLines; i++) { write_hline_lm((x) - 1, (x) - 1 + size, (y) - 1 + size * 3 - i, 1, 1); } } void printTime(uint16_t x, uint16_t y) { char temp[9] = { 0 }; sprintf(temp, "%02d:%02d:%02d", timex.hour, timex.min, timex.sec); // printTextFB(x,y,temp); write_string(temp, x, y, 0, 0, TEXT_VA_TOP, TEXT_HA_LEFT, 0, 3); } /* void drawAltitude(uint16_t x, uint16_t y, int16_t alt, uint8_t dir) { char temp[9]={0}; char updown=' '; uint16_t charx=x/16; if(dir==0) updown=24; if(dir==1) updown=25; sprintf(temp,"%c%6dm",updown,alt); printTextFB(charx,y+2,temp); // frame drawBox(charx*16-3,y,charx*16+strlen(temp)*8+3,y+11); }*/ /** * hud_draw_vertical_scale: Draw a vertical scale. * * @param v value to display as an integer * @param range range about value to display (+/- range/2 each direction) * @param halign horizontal alignment: -1 = left, +1 = right. * @param x x displacement (typ. 0) * @param y y displacement (typ. half display height) * @param height height of scale * @param mintick_step how often a minor tick is shown * @param majtick_step how often a major tick is shown * @param mintick_len minor tick length * @param majtick_len major tick length * @param boundtick_len boundary tick length * @param max_val maximum expected value (used to compute size of arrow ticker) * @param flags special flags (see hud.h.) */ void hud_draw_vertical_scale(int v, int range, int halign, int x, int y, int height, int mintick_step, int majtick_step, int mintick_len, int majtick_len, int boundtick_len, __attribute__((unused)) int max_val, int flags) { char temp[15]; // , temp2[15]; struct FontEntry font_info; struct FontDimensions dim; // Halign should be in a small span. // MY_ASSERT(halign >= -1 && halign <= 1); // Compute the position of the elements. int majtick_start = 0, majtick_end = 0, mintick_start = 0, mintick_end = 0, boundtick_start = 0, boundtick_end = 0; if (halign == -1) { majtick_start = x; majtick_end = x + majtick_len; mintick_start = x; mintick_end = x + mintick_len; boundtick_start = x; boundtick_end = x + boundtick_len; } else if (halign == +1) { x = x - GRAPHICS_HDEADBAND; majtick_start = GRAPHICS_WIDTH_REAL - x - 1; majtick_end = GRAPHICS_WIDTH_REAL - x - majtick_len - 1; mintick_start = GRAPHICS_WIDTH_REAL - x - 1; mintick_end = GRAPHICS_WIDTH_REAL - x - mintick_len - 1; boundtick_start = GRAPHICS_WIDTH_REAL - x - 1; boundtick_end = GRAPHICS_WIDTH_REAL - x - boundtick_len - 1; } // Retrieve width of large font (font #0); from this calculate the x spacing. fetch_font_info(0, 0, &font_info, NULL); int arrow_len = (font_info.height / 2) + 1; // FIXME, font info being loaded correctly?? int text_x_spacing = arrow_len; int max_text_y = 0, text_length = 0; int small_font_char_width = font_info.width + 1; // +1 for horizontal spacing = 1 // For -(range / 2) to +(range / 2), draw the scale. int range_2 = range / 2; // , height_2 = height / 2; int r = 0, rr = 0, rv = 0, ys = 0, style = 0; // calc_ys = 0, // Iterate through each step. for (r = -range_2; r <= +range_2; r++) { style = 0; rr = r + range_2 - v; // normalise range for modulo, subtract value to move ticker tape rv = -rr + range_2; // for number display if (flags & HUD_VSCALE_FLAG_NO_NEGATIVE) { rr += majtick_step / 2; } if (rr % majtick_step == 0) { style = 1; // major tick } else if (rr % mintick_step == 0) { style = 2; // minor tick } else { style = 0; } if (flags & HUD_VSCALE_FLAG_NO_NEGATIVE && rv < 0) { continue; } if (style) { // Calculate y position. ys = ((long int)(r * height) / (long int)range) + y; // sprintf(temp, "ys=%d", ys); // con_puts(temp, 0); // Depending on style, draw a minor or a major tick. if (style == 1) { write_hline_outlined(majtick_start, majtick_end, ys, 2, 2, 0, 1); memset(temp, ' ', 10); // my_itoa(rv, temp); sprintf(temp, "%d", rv); text_length = (strlen(temp) + 1) * small_font_char_width; // add 1 for margin if (text_length > max_text_y) { max_text_y = text_length; } if (halign == -1) { write_string(temp, majtick_end + text_x_spacing, ys, 1, 0, TEXT_VA_MIDDLE, TEXT_HA_LEFT, 0, 1); } else { write_string(temp, majtick_end - text_x_spacing + 1, ys, 1, 0, TEXT_VA_MIDDLE, TEXT_HA_RIGHT, 0, 1); } } else if (style == 2) { write_hline_outlined(mintick_start, mintick_end, ys, 2, 2, 0, 1); } } } // Generate the string for the value, as well as calculating its dimensions. memset(temp, ' ', 10); // my_itoa(v, temp); sprintf(temp, "%d", v); // TODO: add auto-sizing. calc_text_dimensions(temp, font_info, 1, 0, &dim); int xx = 0, i = 0; if (halign == -1) { xx = majtick_end + text_x_spacing; } else { xx = majtick_end - text_x_spacing; } // Draw an arrow from the number to the point. for (i = 0; i < arrow_len; i++) { if (halign == -1) { write_pixel_lm(xx - arrow_len + i, y - i - 1, 1, 1); write_pixel_lm(xx - arrow_len + i, y + i - 1, 1, 1); write_hline_lm(xx + dim.width - 1, xx - arrow_len + i + 1, y - i - 1, 0, 1); write_hline_lm(xx + dim.width - 1, xx - arrow_len + i + 1, y + i - 1, 0, 1); } else { write_pixel_lm(xx + arrow_len - i, y - i - 1, 1, 1); write_pixel_lm(xx + arrow_len - i, y + i - 1, 1, 1); write_hline_lm(xx - dim.width - 1, xx + arrow_len - i - 1, y - i - 1, 0, 1); write_hline_lm(xx - dim.width - 1, xx + arrow_len - i - 1, y + i - 1, 0, 1); } // FIXME // write_hline_lm(xx - dim.width - 1, xx + (arrow_len - i), y - i - 1, 1, 1); // write_hline_lm(xx - dim.width - 1, xx + (arrow_len - i), y + i - 1, 1, 1); } if (halign == -1) { write_hline_lm(xx, xx + dim.width - 1, y - arrow_len, 1, 1); write_hline_lm(xx, xx + dim.width - 1, y + arrow_len - 2, 1, 1); write_vline_lm(xx + dim.width - 1, y - arrow_len, y + arrow_len - 2, 1, 1); } else { write_hline_lm(xx, xx - dim.width - 1, y - arrow_len, 1, 1); write_hline_lm(xx, xx - dim.width - 1, y + arrow_len - 2, 1, 1); write_vline_lm(xx - dim.width - 1, y - arrow_len, y + arrow_len - 2, 1, 1); } // Draw the text. if (halign == -1) { write_string(temp, xx, y, 1, 0, TEXT_VA_MIDDLE, TEXT_HA_LEFT, 0, 0); } else { write_string(temp, xx, y, 1, 0, TEXT_VA_MIDDLE, TEXT_HA_RIGHT, 0, 0); } // Then, add a slow cut off on the edges, so the text doesn't sharply // disappear. We simply clear the areas above and below the ticker, and we // use little markers on the edges. if (halign == -1) { write_filled_rectangle_lm(majtick_end + text_x_spacing, y + (height / 2) - (font_info.height / 2), max_text_y - boundtick_start, font_info.height, 0, 0); write_filled_rectangle_lm(majtick_end + text_x_spacing, y - (height / 2) - (font_info.height / 2), max_text_y - boundtick_start, font_info.height, 0, 0); } else { write_filled_rectangle_lm(majtick_end - text_x_spacing - max_text_y, y + (height / 2) - (font_info.height / 2), max_text_y, font_info.height, 0, 0); write_filled_rectangle_lm(majtick_end - text_x_spacing - max_text_y, y - (height / 2) - (font_info.height / 2), max_text_y, font_info.height, 0, 0); } write_hline_outlined(boundtick_start, boundtick_end, y + (height / 2), 2, 2, 0, 1); write_hline_outlined(boundtick_start, boundtick_end, y - (height / 2), 2, 2, 0, 1); } /** * hud_draw_compass: Draw a compass. * * @param v value for the compass * @param range range about value to display (+/- range/2 each direction) * @param width length in pixels * @param x x displacement (typ. half display width) * @param y y displacement (typ. bottom of display) * @param mintick_step how often a minor tick is shown * @param majtick_step how often a major tick (heading "xx") is shown * @param mintick_len minor tick length * @param majtick_len major tick length * @param flags special flags (see hud.h.) */ void hud_draw_linear_compass(int v, int range, int width, int x, int y, int mintick_step, int majtick_step, int mintick_len, int majtick_len, __attribute__((unused)) int flags) { v %= 360; // wrap, just in case. struct FontEntry font_info; int majtick_start = 0, majtick_end = 0, mintick_start = 0, mintick_end = 0, textoffset = 0; char headingstr[4]; majtick_start = y; majtick_end = y - majtick_len; mintick_start = y; mintick_end = y - mintick_len; textoffset = 8; int r, style, rr, xs; // rv, int range_2 = range / 2; for (r = -range_2; r <= +range_2; r++) { style = 0; rr = (v + r + 360) % 360; // normalise range for modulo, add to move compass track // rv = -rr + range_2; // for number display if (rr % majtick_step == 0) { style = 1; // major tick } else if (rr % mintick_step == 0) { style = 2; // minor tick } if (style) { // Calculate x position. xs = ((long int)(r * width) / (long int)range) + x; // Draw it. if (style == 1) { write_vline_outlined(xs, majtick_start, majtick_end, 2, 2, 0, 1); // Draw heading above this tick. // If it's not one of north, south, east, west, draw the heading. // Otherwise, draw one of the identifiers. if (rr % 90 != 0) { // We abbreviate heading to two digits. This has the side effect of being easy to compute. headingstr[0] = '0' + (rr / 100); headingstr[1] = '0' + ((rr / 10) % 10); headingstr[2] = 0; headingstr[3] = 0; // nul to terminate } else { switch (rr) { case 0: headingstr[0] = 'N'; break; case 90: headingstr[0] = 'E'; break; case 180: headingstr[0] = 'S'; break; case 270: headingstr[0] = 'W'; break; } headingstr[1] = 0; headingstr[2] = 0; headingstr[3] = 0; } // +1 fudge...! write_string(headingstr, xs + 1, majtick_start + textoffset, 1, 0, TEXT_VA_MIDDLE, TEXT_HA_CENTER, 0, 1); } else if (style == 2) { write_vline_outlined(xs, mintick_start, mintick_end, 2, 2, 0, 1); } } } // Then, draw a rectangle with the present heading in it. // We want to cover up any other markers on the bottom. // First compute font size. fetch_font_info(0, 3, &font_info, NULL); int text_width = (font_info.width + 1) * 3; int rect_width = text_width + 2; write_filled_rectangle_lm(x - (rect_width / 2), majtick_start + 2, rect_width, font_info.height + 2, 0, 1); write_rectangle_outlined(x - (rect_width / 2), majtick_start + 2, rect_width, font_info.height + 2, 0, 1); headingstr[0] = '0' + (v / 100); headingstr[1] = '0' + ((v / 10) % 10); headingstr[2] = '0' + (v % 10); headingstr[3] = 0; write_string(headingstr, x + 1, majtick_start + textoffset + 2, 0, 0, TEXT_VA_MIDDLE, TEXT_HA_CENTER, 1, 3); } // CORE draw routines end here void draw_artificial_horizon(float angle, float pitch, int16_t l_x, int16_t l_y, int16_t size) { float alpha; uint8_t vertical = 0, horizontal = 0; int16_t x1, x2; int16_t y1, y2; int16_t refx, refy; alpha = DEG2RAD(angle); refx = l_x + size / 2; refy = l_y + size / 2; // float k = 0; float dx = sinf(alpha) * (pitch / 90.0f * (size / 2)); float dy = cosf(alpha) * (pitch / 90.0f * (size / 2)); int16_t x0 = (size / 2) - dx; int16_t y0 = (size / 2) + dy; // calculate the line function if ((angle < 90.0f) && (angle > -90)) { vertical = 0; if (fabsf(angle) < 1e-5f) { horizontal = 1; } else { k = tanf(alpha); } } else { vertical = 1; } // crossing point of line if (!vertical && !horizontal) { // y-y0=k(x-x0) int16_t x = 0; int16_t y = k * (x - x0) + y0; // find right crossing point x1 = x; y1 = y; if (y < 0) { y1 = 0; x1 = ((y1 - y0) + k * x0) / k; } if (y > size) { y1 = size; x1 = ((y1 - y0) + k * x0) / k; } // left crossing point x = size; y = k * (x - x0) + y0; x2 = x; y2 = y; if (y < 0) { y2 = 0; x2 = ((y2 - y0) + k * x0) / k; } if (y > size) { y2 = size; x2 = ((y2 - y0) + k * x0) / k; } // move to location // horizon line write_line_outlined(x1 + l_x, y1 + l_y, x2 + l_x, y2 + l_y, 0, 0, 0, 1); // fill if (angle <= 0.0f && angle > -90.0f) { // write_string("1", APPLY_HDEADBAND((GRAPHICS_RIGHT/2)),APPLY_VDEADBAND(GRAPHICS_BOTTOM-10), 0, 0, TEXT_VA_BOTTOM, TEXT_HA_CENTER, 0, 3); for (int i = y2; i < size; i++) { x2 = ((i - y0) + k * x0) / k; if (x2 > size) { x2 = size; } if (x2 < 0) { x2 = 0; } write_hline_lm(x2 + l_x, size + l_x, i + l_y, 1, 1); } } else if (angle < -90.0f) { // write_string("2", APPLY_HDEADBAND((GRAPHICS_RIGHT/2)),APPLY_VDEADBAND(GRAPHICS_BOTTOM-10), 0, 0, TEXT_VA_BOTTOM, TEXT_HA_CENTER, 0, 3); for (int i = 0; i < y2; i++) { x2 = ((i - y0) + k * x0) / k; if (x2 > size) { x2 = size; } if (x2 < 0) { x2 = 0; } write_hline_lm(size + l_x, x2 + l_x, i + l_y, 1, 1); } } else if (angle > 0.0f && angle < 90.0f) { // write_string("3", APPLY_HDEADBAND((GRAPHICS_RIGHT/2)),APPLY_VDEADBAND(GRAPHICS_BOTTOM-10), 0, 0, TEXT_VA_BOTTOM, TEXT_HA_CENTER, 0, 3); for (int i = y1; i < size; i++) { x2 = ((i - y0) + k * x0) / k; if (x2 > size) { x2 = size; } if (x2 < 0) { x2 = 0; } write_hline_lm(0 + l_x, x2 + l_x, i + l_y, 1, 1); } } else if (angle > 90.0f) { // write_string("4", APPLY_HDEADBAND((GRAPHICS_RIGHT/2)),APPLY_VDEADBAND(GRAPHICS_BOTTOM-10), 0, 0, TEXT_VA_BOTTOM, TEXT_HA_CENTER, 0, 3); for (int i = 0; i < y1; i++) { x2 = ((i - y0) + k * x0) / k; if (x2 > size) { x2 = size; } if (x2 < 0) { x2 = 0; } write_hline_lm(x2 + l_x, 0 + l_x, i + l_y, 1, 1); } } } else if (vertical) { // horizon line write_line_outlined(x0 + l_x, 0 + l_y, x0 + l_x, size + l_y, 0, 0, 0, 1); if (angle >= 90.0f) { // write_string("5", APPLY_HDEADBAND((GRAPHICS_RIGHT/2)),APPLY_VDEADBAND(GRAPHICS_BOTTOM-10), 0, 0, TEXT_VA_BOTTOM, TEXT_HA_CENTER, 0, 3); for (int i = 0; i < size; i++) { write_hline_lm(0 + l_x, x0 + l_x, i + l_y, 1, 1); } } else { // write_string("6", APPLY_HDEADBAND((GRAPHICS_RIGHT/2)),APPLY_VDEADBAND(GRAPHICS_BOTTOM-10), 0, 0, TEXT_VA_BOTTOM, TEXT_HA_CENTER, 0, 3); for (int i = 0; i < size; i++) { write_hline_lm(size + l_x, x0 + l_x, i + l_y, 1, 1); } } } else if (horizontal) { // horizon line write_hline_outlined(0 + l_x, size + l_x, y0 + l_y, 0, 0, 0, 1); if (angle < 0) { // write_string("7", APPLY_HDEADBAND((GRAPHICS_RIGHT/2)),APPLY_VDEADBAND(GRAPHICS_BOTTOM-10), 0, 0, TEXT_VA_BOTTOM, TEXT_HA_CENTER, 0, 3); for (int i = 0; i < y0; i++) { write_hline_lm(0 + l_x, size + l_x, i + l_y, 1, 1); } } else { // write_string("8", APPLY_HDEADBAND((GRAPHICS_RIGHT/2)),APPLY_VDEADBAND(GRAPHICS_BOTTOM-10), 0, 0, TEXT_VA_BOTTOM, TEXT_HA_CENTER, 0, 3); for (int i = y0; i < size; i++) { write_hline_lm(0 + l_x, size + l_x, i + l_y, 1, 1); } } } // sides write_line_outlined(l_x, l_y, l_x, l_y + size, 0, 0, 0, 1); write_line_outlined(l_x + size, l_y, l_x + size, l_y + size, 0, 0, 0, 1); // plane write_line_outlined(refx - 5, refy, refx + 6, refy, 0, 0, 0, 1); write_line_outlined(refx, refy, refx, refy - 3, 0, 0, 0, 1); } void introText() { write_string("ver 0.2", APPLY_HDEADBAND((GRAPHICS_RIGHT / 2)), APPLY_VDEADBAND(GRAPHICS_BOTTOM - 10), 0, 0, TEXT_VA_BOTTOM, TEXT_HA_CENTER, 0, 3); } void introGraphics() { /* logo */ int image = 0; struct splashEntry splash_info; splash_info = splash[image]; copyimage(APPLY_HDEADBAND(GRAPHICS_RIGHT / 2 - (splash_info.width) / 2), APPLY_VDEADBAND(GRAPHICS_BOTTOM / 2 - (splash_info.height) / 2), image); /* frame */ drawBox(APPLY_HDEADBAND(0), APPLY_VDEADBAND(0), APPLY_HDEADBAND(GRAPHICS_RIGHT - 8), APPLY_VDEADBAND(GRAPHICS_BOTTOM)); // Must mask out last half-word because SPI keeps clocking it out otherwise for (uint32_t i = 0; i < 8; i++) { write_vline(draw_buffer_level, GRAPHICS_WIDTH_REAL - i - 1, 0, GRAPHICS_HEIGHT_REAL - 1, 0); write_vline(draw_buffer_mask, GRAPHICS_WIDTH_REAL - i - 1, 0, GRAPHICS_HEIGHT_REAL - 1, 0); } } void calcHomeArrow(int16_t m_yaw) { HomeLocationData home; HomeLocationGet(&home); GPSPositionSensorData gpsData; GPSPositionSensorGet(&gpsData); /** http://www.movable-type.co.uk/scripts/latlong.html **/ float lat1, lat2, lon1, lon2, a, c, d, x, y, brng, u2g; float elevation; float gcsAlt = home.Altitude; // Home MSL altitude float uavAlt = gpsData.Altitude; // UAV MSL altitude float dAlt = uavAlt - gcsAlt; // Altitude difference // Convert to radians lat1 = DEG2RAD(home.Latitude) / 10000000.0f; // Home lat lon1 = DEG2RAD(home.Longitude) / 10000000.0f; // Home lon lat2 = DEG2RAD(gpsData.Latitude) / 10000000.0f; // UAV lat lon2 = DEG2RAD(gpsData.Longitude) / 10000000.0f; // UAV lon // Bearing /** var y = Math.sin(dLon) * Math.cos(lat2); var x = Math.cos(lat1)*Math.sin(lat2) - Math.sin(lat1)*Math.cos(lat2)*Math.cos(dLon); var brng = Math.atan2(y, x).toDeg(); **/ y = sinf(lon2 - lon1) * cosf(lat2); x = cosf(lat1) * sinf(lat2) - sinf(lat1) * cosf(lat2) * cosf(lon2 - lon1); brng = RAD2DEG(atan2f(y, x)); if (brng < 0) { brng += 360.0f; } // yaw corrected bearing, needs compass u2g = brng - 180.0f - m_yaw; if (u2g < 0) { u2g += 360.0f; } // Haversine formula for distance /** var R = 6371; // km var dLat = (lat2-lat1).toRad(); var dLon = (lon2-lon1).toRad(); var a = Math.sin(dLat/2) * Math.sin(dLat/2) + Math.cos(lat1.toRad()) * Math.cos(lat2.toRad()) * Math.sin(dLon/2) * Math.sin(dLon/2); var c = 2 * Math.atan2(Math.sqrt(a), Math.sqrt(1-a)); var d = R * c; **/ a = sinf((lat2 - lat1) / 2) * sinf((lat2 - lat1) / 2) + cosf(lat1) * cosf(lat2) * sinf((lon2 - lon1) / 2) * sinf((lon2 - lon1) / 2); c = 2.0f * atan2f(sqrtf(a), sqrtf(1.0f - a)); d = 6371.0f * 1000.0f * c; // Elevation v depends servo direction if (d > 0.0f) { elevation = 90.0f - RAD2DEG(atanf(dAlt / d)); } else { elevation = 0.0f; } // ! TODO: sanity check char temp[50] = { 0 }; sprintf(temp, "hea:%d", (int)brng); write_string(temp, APPLY_HDEADBAND(GRAPHICS_RIGHT / 2 - 30), APPLY_VDEADBAND(30), 0, 0, TEXT_VA_TOP, TEXT_HA_LEFT, 0, 2); sprintf(temp, "ele:%d", (int)elevation); write_string(temp, APPLY_HDEADBAND(GRAPHICS_RIGHT / 2 - 30), APPLY_VDEADBAND(30 + 10), 0, 0, TEXT_VA_TOP, TEXT_HA_LEFT, 0, 2); sprintf(temp, "dis:%d", (int)d); write_string(temp, APPLY_HDEADBAND(GRAPHICS_RIGHT / 2 - 30), APPLY_VDEADBAND(30 + 10 + 10), 0, 0, TEXT_VA_TOP, TEXT_HA_LEFT, 0, 2); sprintf(temp, "u2g:%d", (int)u2g); write_string(temp, APPLY_HDEADBAND(GRAPHICS_RIGHT / 2 - 30), APPLY_VDEADBAND(30 + 10 + 10 + 10), 0, 0, TEXT_VA_TOP, TEXT_HA_LEFT, 0, 2); sprintf(temp, "%c%c", (int)(u2g / 22.5f) * 2 + 0x90, (int)(u2g / 22.5f) * 2 + 0x91); write_string(temp, APPLY_HDEADBAND(250), APPLY_VDEADBAND(40 + 10 + 10), 0, 0, TEXT_VA_TOP, TEXT_HA_LEFT, 0, 3); } int lama = 10; int lama_loc[2][30]; void lamas(void) { char temp[10] = { 0 }; lama++; if (lama % 10 == 0) { for (int z = 0; z < 30; z++) { lama_loc[0][z] = rand() % (GRAPHICS_RIGHT - 10); lama_loc[1][z] = rand() % (GRAPHICS_BOTTOM - 10); } } for (int z = 0; z < 30; z++) { sprintf(temp, "%c", 0xe8 + (lama_loc[0][z] % 2)); write_string(temp, APPLY_HDEADBAND(lama_loc[0][z]), APPLY_VDEADBAND(lama_loc[1][z]), 0, 0, TEXT_VA_TOP, TEXT_HA_LEFT, 0, 2); } } // main draw function void updateGraphics() { OsdSettingsData OsdSettings; OsdSettingsGet(&OsdSettings); AttitudeStateData attitude; AttitudeStateGet(&attitude); GPSPositionSensorData gpsData; GPSPositionSensorGet(&gpsData); HomeLocationData home; HomeLocationGet(&home); BaroSensorData baro; BaroSensorGet(&baro); FlightStatusData status; FlightStatusGet(&status); PIOS_Servo_Set(0, OsdSettings.White); PIOS_Servo_Set(1, OsdSettings.Black); switch (OsdSettings.Screen) { case 0: // Dave simple { if (home.Set == HOMELOCATION_SET_FALSE) { char temps[20] = { 0 }; sprintf(temps, "HOME NOT SET"); // printTextFB(x,y,temp); write_string(temps, APPLY_HDEADBAND(GRAPHICS_RIGHT / 2), (GRAPHICS_BOTTOM / 2), 0, 0, TEXT_VA_TOP, TEXT_HA_CENTER, 0, 3); } char temp[50] = { 0 }; memset(temp, ' ', 40); // Note: cast to double required due to -Wdouble-promotion compiler option is // being used, and there is no way in C to pass a float to a variadic function like sprintf() sprintf(temp, "Lat:%11.7f", (double)(gpsData.Latitude / 10000000.0f)); write_string(temp, APPLY_HDEADBAND(20), APPLY_VDEADBAND(GRAPHICS_BOTTOM - 30), 0, 0, TEXT_VA_BOTTOM, TEXT_HA_LEFT, 0, 3); sprintf(temp, "Lon:%11.7f", (double)(gpsData.Longitude / 10000000.0f)); write_string(temp, APPLY_HDEADBAND(20), APPLY_VDEADBAND(GRAPHICS_BOTTOM - 10), 0, 0, TEXT_VA_BOTTOM, TEXT_HA_LEFT, 0, 3); sprintf(temp, "Sat:%d", (int)gpsData.Satellites); write_string(temp, APPLY_HDEADBAND(GRAPHICS_RIGHT - 40), APPLY_VDEADBAND(30), 0, 0, TEXT_VA_TOP, TEXT_HA_RIGHT, 0, 2); /* Print ADC voltage FLIGHT*/ sprintf(temp, "V:%5.2fV", (double)(PIOS_ADC_PinGet(2) * 3 * 6.1f / 4096)); write_string(temp, APPLY_HDEADBAND(20), APPLY_VDEADBAND(20), 0, 0, TEXT_VA_TOP, TEXT_HA_LEFT, 0, 3); if (gpsData.Heading > 180) { calcHomeArrow((int16_t)(gpsData.Heading - 360)); } else { calcHomeArrow((int16_t)(gpsData.Heading)); } } break; case 1: { /*drawBox(2,2,GRAPHICS_WIDTH_REAL-4,GRAPHICS_HEIGHT_REAL-4); write_filled_rectangle(draw_buffer_mask,0,0,GRAPHICS_WIDTH_REAL-2,GRAPHICS_HEIGHT_REAL-2,0); write_filled_rectangle(draw_buffer_mask,2,2,GRAPHICS_WIDTH_REAL-4-2,GRAPHICS_HEIGHT_REAL-4-2,2); write_filled_rectangle(draw_buffer_mask,3,3,GRAPHICS_WIDTH_REAL-4-1,GRAPHICS_HEIGHT_REAL-4-1,0);*/ // write_filled_rectangle(draw_buffer_mask,5,5,GRAPHICS_WIDTH_REAL-4-5,GRAPHICS_HEIGHT_REAL-4-5,0); // write_rectangle_outlined(10,10,GRAPHICS_WIDTH_REAL-20,GRAPHICS_HEIGHT_REAL-20,0,0); // drawLine(GRAPHICS_WIDTH_REAL-1, GRAPHICS_HEIGHT_REAL-1,(GRAPHICS_WIDTH_REAL/2)-1, GRAPHICS_HEIGHT_REAL-1 ); // drawCircle((GRAPHICS_WIDTH_REAL/2)-1, (GRAPHICS_HEIGHT_REAL/2)-1, (GRAPHICS_HEIGHT_REAL/2)-1); // drawCircle((GRAPHICS_SIZE/2)-1, (GRAPHICS_SIZE/2)-1, (GRAPHICS_SIZE/2)-2); // drawLine(0, (GRAPHICS_SIZE/2)-1, GRAPHICS_SIZE-1, (GRAPHICS_SIZE/2)-1); // drawLine((GRAPHICS_SIZE/2)-1, 0, (GRAPHICS_SIZE/2)-1, GRAPHICS_SIZE-1); /*angleA++; if(angleB<=-90) { sum=2; } if(angleB>=90) { sum=-2; } angleB+=sum; angleC+=2;*/ // GPS HACK if (gpsData.Heading > 180) { calcHomeArrow((int16_t)(gpsData.Heading - 360)); } else { calcHomeArrow((int16_t)(gpsData.Heading)); } /* Draw Attitude Indicator */ if (OsdSettings.Attitude == OSDSETTINGS_ATTITUDE_ENABLED) { drawAttitude(APPLY_HDEADBAND(OsdSettings.AttitudeSetup.X), APPLY_VDEADBAND(OsdSettings.AttitudeSetup.Y), attitude.Pitch, attitude.Roll, 96); } // write_string("Hello OP-OSD", 60, 12, 1, 0, TEXT_VA_TOP, TEXT_HA_LEFT, 0, 0); // printText16( 60, 12,"Hello OP-OSD"); char temp[50] = { 0 }; memset(temp, ' ', 40); sprintf(temp, "Lat:%11.7f", (double)(gpsData.Latitude / 10000000.0f)); write_string(temp, APPLY_HDEADBAND(5), APPLY_VDEADBAND(5), 0, 0, TEXT_VA_TOP, TEXT_HA_LEFT, 0, 2); sprintf(temp, "Lon:%11.7f", (double)(gpsData.Longitude / 10000000.0f)); write_string(temp, APPLY_HDEADBAND(5), APPLY_VDEADBAND(15), 0, 0, TEXT_VA_TOP, TEXT_HA_LEFT, 0, 2); sprintf(temp, "Fix:%d", (int)gpsData.Status); write_string(temp, APPLY_HDEADBAND(5), APPLY_VDEADBAND(25), 0, 0, TEXT_VA_TOP, TEXT_HA_LEFT, 0, 2); sprintf(temp, "Sat:%d", (int)gpsData.Satellites); write_string(temp, APPLY_HDEADBAND(5), APPLY_VDEADBAND(35), 0, 0, TEXT_VA_TOP, TEXT_HA_LEFT, 0, 2); /* Print RTC time */ if (OsdSettings.Time == OSDSETTINGS_TIME_ENABLED) { printTime(APPLY_HDEADBAND(OsdSettings.TimeSetup.X), APPLY_VDEADBAND(OsdSettings.TimeSetup.Y)); } /* Print Number of detected video Lines */ sprintf(temp, "Lines:%4d", PIOS_Video_GetOSDLines()); write_string(temp, APPLY_HDEADBAND((GRAPHICS_RIGHT - 8)), APPLY_VDEADBAND(5), 0, 0, TEXT_VA_TOP, TEXT_HA_RIGHT, 0, 2); /* Print ADC voltage */ // sprintf(temp,"Rssi:%4dV",(int)(PIOS_ADC_PinGet(4)*3000/4096)); // write_string(temp, (GRAPHICS_WIDTH_REAL - 2),15, 0, 0, TEXT_VA_TOP, TEXT_HA_RIGHT, 0, 2); sprintf(temp, "Rssi:%4.2fV", (double)(PIOS_ADC_PinGet(5) * 3.0f / 4096.0f)); write_string(temp, APPLY_HDEADBAND((GRAPHICS_RIGHT - 8)), APPLY_VDEADBAND(15), 0, 0, TEXT_VA_TOP, TEXT_HA_RIGHT, 0, 2); /* Print CPU temperature */ sprintf(temp, "Temp:%4.2fC", (double)(PIOS_ADC_PinGet(3) * 0.29296875f - 264)); write_string(temp, APPLY_HDEADBAND((GRAPHICS_RIGHT - 8)), APPLY_VDEADBAND(25), 0, 0, TEXT_VA_TOP, TEXT_HA_RIGHT, 0, 2); /* Print ADC voltage FLIGHT*/ sprintf(temp, "FltV:%4.2fV", (double)(PIOS_ADC_PinGet(2) * 3.0f * 6.1f / 4096.0f)); write_string(temp, APPLY_HDEADBAND((GRAPHICS_RIGHT - 8)), APPLY_VDEADBAND(35), 0, 0, TEXT_VA_TOP, TEXT_HA_RIGHT, 0, 2); /* Print ADC voltage VIDEO*/ sprintf(temp, "VidV:%4.2fV", (double)(PIOS_ADC_PinGet(4) * 3.0f * 6.1f / 4096.0f)); write_string(temp, APPLY_HDEADBAND((GRAPHICS_RIGHT - 8)), APPLY_VDEADBAND(45), 0, 0, TEXT_VA_TOP, TEXT_HA_RIGHT, 0, 2); /* Print ADC voltage RSSI */ // sprintf(temp,"Curr:%4dA",(int)(PIOS_ADC_PinGet(0)*300*61/4096)); // write_string(temp, (GRAPHICS_WIDTH_REAL - 2),60, 0, 0, TEXT_VA_TOP, TEXT_HA_RIGHT, 0, 2); /* Draw Battery Gauge */ /*m_batt++; uint8_t dir=3; if(m_batt==101) m_batt=0; if(m_pitch>0) { dir=0; m_alt+=m_pitch/2; } else if(m_pitch<0) { dir=1; m_alt+=m_pitch/2; }*/ /*if(OsdSettings.Battery == OSDSETTINGS_BATTERY_ENABLED) { drawBattery(APPLY_HDEADBAND(OsdSettings.BatterySetup[OSDSETTINGS_BATTERYSETUP_X]),APPLY_VDEADBAND(OsdSettings.BatterySetup[OSDSETTINGS_BATTERYSETUP_Y]),m_batt,16); }*/ // drawAltitude(200,50,m_alt,dir); // drawArrow(96,GRAPHICS_HEIGHT_REAL/2,angleB,32); // Draw airspeed (left side.) if (OsdSettings.Speed == OSDSETTINGS_SPEED_ENABLED) { hud_draw_vertical_scale((int)gpsData.Groundspeed, 100, -1, APPLY_HDEADBAND(OsdSettings.SpeedSetup.X), APPLY_VDEADBAND(OsdSettings.SpeedSetup.Y), 100, 10, 20, 7, 12, 15, 1000, HUD_VSCALE_FLAG_NO_NEGATIVE); } // Draw altimeter (right side.) if (OsdSettings.Altitude == OSDSETTINGS_ALTITUDE_ENABLED) { hud_draw_vertical_scale((int)gpsData.Altitude, 200, +1, APPLY_HDEADBAND(OsdSettings.AltitudeSetup.X), APPLY_VDEADBAND(OsdSettings.AltitudeSetup.Y), 100, 20, 100, 7, 12, 15, 500, 0); } // Draw compass. if (OsdSettings.Heading == OSDSETTINGS_HEADING_ENABLED) { if (attitude.Yaw < 0) { hud_draw_linear_compass(360 + attitude.Yaw, 150, 120, APPLY_HDEADBAND(OsdSettings.HeadingSetup.X), APPLY_VDEADBAND(OsdSettings.HeadingSetup.Y), 15, 30, 7, 12, 0); } else { hud_draw_linear_compass(attitude.Yaw, 150, 120, APPLY_HDEADBAND(OsdSettings.HeadingSetup.X), APPLY_VDEADBAND(OsdSettings.HeadingSetup.Y), 15, 30, 7, 12, 0); } } } break; case 2: { int size = 64; int x = ((GRAPHICS_RIGHT / 2) - (size / 2)), y = (GRAPHICS_BOTTOM - size - 2); draw_artificial_horizon(-attitude.Roll, attitude.Pitch, APPLY_HDEADBAND(x), APPLY_VDEADBAND(y), size); hud_draw_vertical_scale((int)gpsData.Groundspeed, 20, +1, APPLY_HDEADBAND(GRAPHICS_RIGHT - (x - 1)), APPLY_VDEADBAND(y + (size / 2)), size, 5, 10, 4, 7, 10, 100, HUD_VSCALE_FLAG_NO_NEGATIVE); if (OsdSettings.AltitudeSource == OSDSETTINGS_ALTITUDESOURCE_BARO) { hud_draw_vertical_scale((int)baro.Altitude, 50, -1, APPLY_HDEADBAND((x + size + 1)), APPLY_VDEADBAND(y + (size / 2)), size, 10, 20, 4, 7, 10, 500, 0); } else { hud_draw_vertical_scale((int)gpsData.Altitude, 50, -1, APPLY_HDEADBAND((x + size + 1)), APPLY_VDEADBAND(y + (size / 2)), size, 10, 20, 4, 7, 10, 500, 0); } char temp[50] = { 0 }; memset(temp, ' ', 50); switch (status.FlightMode) { case FLIGHTSTATUS_FLIGHTMODE_MANUAL: sprintf(temp, "Man"); break; case FLIGHTSTATUS_FLIGHTMODE_STABILIZED1: sprintf(temp, "Stab1"); break; case FLIGHTSTATUS_FLIGHTMODE_STABILIZED2: sprintf(temp, "Stab2"); break; case FLIGHTSTATUS_FLIGHTMODE_STABILIZED3: sprintf(temp, "Stab3"); break; case FLIGHTSTATUS_FLIGHTMODE_POSITIONHOLD: sprintf(temp, "PH"); break; case FLIGHTSTATUS_FLIGHTMODE_RETURNTOBASE: sprintf(temp, "RTB"); break; case FLIGHTSTATUS_FLIGHTMODE_PATHPLANNER: sprintf(temp, "PATH"); break; default: sprintf(temp, "Mode: %d", status.FlightMode); break; } write_string(temp, APPLY_HDEADBAND(5), APPLY_VDEADBAND(5), 0, 0, TEXT_VA_TOP, TEXT_HA_LEFT, 0, 2); } break; case 3: { lamas(); } break; case 4: case 5: case 6: { int image = OsdSettings.Screen - 4; struct splashEntry splash_info; splash_info = splash[image]; copyimage(APPLY_HDEADBAND(GRAPHICS_RIGHT / 2 - (splash_info.width) / 2), APPLY_VDEADBAND(GRAPHICS_BOTTOM / 2 - (splash_info.height) / 2), image); } break; default: write_vline_lm(APPLY_HDEADBAND(GRAPHICS_RIGHT / 2), APPLY_VDEADBAND(0), APPLY_VDEADBAND(GRAPHICS_BOTTOM), 1, 1); write_hline_lm(APPLY_HDEADBAND(0), APPLY_HDEADBAND(GRAPHICS_RIGHT), APPLY_VDEADBAND(GRAPHICS_BOTTOM / 2), 1, 1); break; } // Must mask out last half-word because SPI keeps clocking it out otherwise for (uint32_t i = 0; i < 8; i++) { write_vline(draw_buffer_level, GRAPHICS_WIDTH_REAL - i - 1, 0, GRAPHICS_HEIGHT_REAL - 1, 0); write_vline(draw_buffer_mask, GRAPHICS_WIDTH_REAL - i - 1, 0, GRAPHICS_HEIGHT_REAL - 1, 0); } } void updateOnceEveryFrame() { clearGraphics(); updateGraphics(); } // **************** /** * Initialise the gps module * \return -1 if initialisation failed * \return 0 on success */ int32_t osdgenStart(void) { // Start gps task vSemaphoreCreateBinary(osdSemaphore); xTaskCreate(osdgenTask, (const char *)"OSDGEN", STACK_SIZE_BYTES / 4, NULL, TASK_PRIORITY, &osdgenTaskHandle); PIOS_TASK_MONITOR_RegisterTask(TASKINFO_RUNNING_OSDGEN, osdgenTaskHandle); #ifdef PIOS_INCLUDE_WDG PIOS_WDG_RegisterFlag(PIOS_WDG_OSDGEN); #endif return 0; } /** * Initialise the osd module * \return -1 if initialisation failed * \return 0 on success */ int32_t osdgenInitialize(void) { AttitudeStateInitialize(); #ifdef PIOS_INCLUDE_GPS GPSPositionSensorInitialize(); #if !defined(PIOS_GPS_MINIMAL) GPSTimeInitialize(); GPSSatellitesInitialize(); #endif #ifdef PIOS_GPS_SETS_HOMELOCATION HomeLocationInitialize(); #endif #endif OsdSettingsInitialize(); BaroSensorInitialize(); FlightStatusInitialize(); return 0; } MODULE_INITCALL(osdgenInitialize, osdgenStart); // **************** /** * Main osd task. It does not return. */ static void osdgenTask(__attribute__((unused)) void *parameters) { // portTickType lastSysTime; // Loop forever // lastSysTime = xTaskGetTickCount(); OsdSettingsData OsdSettings; OsdSettingsGet(&OsdSettings); PIOS_Servo_Set(0, OsdSettings.White); PIOS_Servo_Set(1, OsdSettings.Black); // intro for (int i = 0; i < 63; i++) { if (xSemaphoreTake(osdSemaphore, LONG_TIME) == pdTRUE) { #ifdef PIOS_INCLUDE_WDG PIOS_WDG_UpdateFlag(PIOS_WDG_OSDGEN); #endif clearGraphics(); introGraphics(); } } for (int i = 0; i < 63; i++) { if (xSemaphoreTake(osdSemaphore, LONG_TIME) == pdTRUE) { #ifdef PIOS_INCLUDE_WDG PIOS_WDG_UpdateFlag(PIOS_WDG_OSDGEN); #endif clearGraphics(); introGraphics(); introText(); } } while (1) { if (xSemaphoreTake(osdSemaphore, LONG_TIME) == pdTRUE) { #ifdef PIOS_INCLUDE_WDG PIOS_WDG_UpdateFlag(PIOS_WDG_OSDGEN); #endif updateOnceEveryFrame(); } // xSemaphoreTake(osdSemaphore, portMAX_DELAY); // vTaskDelayUntil(&lastSysTime, 10 / portTICK_RATE_MS); } } // **************** /** * @} * @} */