Subversion Repositories Projects

Rev

Rev 766 | Rev 783 | Go to most recent revision | Details | Compare with Previous | Last modification | View Log | RSS feed

Rev Author Line No. Line
459 cascade 1
/****************************************************************************
728 cascade 2
 *   Copyright (C) 2009-2010 by Claas Anders "CaScAdE" Rathje               *
459 cascade 3
 *   admiralcascade@gmail.com                                               *
4
 *   Project-URL: http://www.mylifesucks.de/oss/c-osd/                      *
5
 *                                                                          *
6
 *   This program is free software; you can redistribute it and/or modify   *
7
 *   it under the terms of the GNU General Public License as published by   *
8
 *   the Free Software Foundation; either version 2 of the License.         *
9
 *                                                                          *
10
 *   This program is distributed in the hope that it will be useful,        *
11
 *   but WITHOUT ANY WARRANTY; without even the implied warranty of         *
12
 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the          *
13
 *   GNU General Public License for more details.                           *
14
 *                                                                          *
15
 *   You should have received a copy of the GNU General Public License      *
16
 *   along with this program; if not, write to the                          *
17
 *   Free Software Foundation, Inc.,                                        *
18
 *   59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.              *
19
 ****************************************************************************/
497 cascade 20
 
21
#include "main.h"
22
#include "max7456_software_spi.h"
23
#include "osd_helpers.h"
24
#include "osd_ncmode_default.h"
25
 
26
#if !(ALLCHARSDEBUG|(WRITECHARS != -1))
27
 
28
int osd_ncmode_default() {
761 - 29
    uint8_t rc_signal = naviData.RC_RSSI ? naviData.RC_RSSI : naviData.RC_Quality; // if RSSI is present use it, else use Qality
753 cascade 30
 
761 - 31
    if (COSD_FLAGS_MODES & COSD_FLAG_HUD) {
32
        // write icons at init or after menu/mode-switch
33
        if (!(COSD_FLAGS_RUNTIME & COSD_ICONS_WRITTEN)) {
34
            if (COSD_FLAGS_CONFIG & COSD_FLAG_FEET) {
35
                write_char_xy(5, top_line, 0x7D); // mph
36
                write_char_xy(27, top_line + 1, 0x7E); // small feet ft home
37
                write_char_xy(27, top_line, 0x7E); // small feet ft height
38
            } else {
39
                write_char_xy(5, top_line, 0xCB); // km/h
40
                write_char_xy(27, top_line + 1, 0xCC); // small meters m home
41
                write_char_xy(27, top_line, 0xCC); // small meters m height
42
            }
757 cascade 43
 
761 - 44
            write_char_xy(10, top_line, 0xCA); // RC-transmitter
45
            write_char_xy(16, top_line, 0xD0); // degree symbol
757 cascade 46
 
761 - 47
            write_char_xy(20, top_line + 1, 0xB0); // left circle
48
            write_char_xy(22, top_line + 1, 0xB2); // right circle
459 cascade 49
 
50
 
761 - 51
            write_char_xy(7, bottom_line, 0x9E); // small V
52
            if ((COSD_FLAGS_RUNTIME & COSD_FLAG_STROMREC) || (COSD_FLAGS_MODES & COSD_FLAG_FCCURRENT)) {
53
                write_char_xy(7, bottom_line - 1, 0x9F); // small A
54
                write_char_xy(14, bottom_line - 1, 0xB5); // mah
55
                if (COSD_FLAGS_MODES & COSD_FLAG_STROMVOLT) {
56
                    write_char_xy(21, bottom_line - 1, 0x9E); // small V
57
                }
58
            }
59
            write_char_xy(14, bottom_line, 0xD1); // on clock
60
            write_char_xy(21, bottom_line, 0xD2); // fly clock
61
            write_char_xy(26, bottom_line, 0xC8); // sat1
62
            write_char_xy(27, bottom_line, 0xC9); // sat2
63
            COSD_FLAGS_RUNTIME |= COSD_ICONS_WRITTEN;
64
        }
459 cascade 65
 
761 - 66
        // first line
67
        if (COSD_FLAGS_CONFIG & COSD_FLAG_FEET) {
68
            // experimental cm/s -> mph
69
            write_ndigit_number_u(2, top_line, (uint16_t)(((uint32_t)naviData.GroundSpeed * (uint32_t)279) / (uint32_t)12500), 3, 0);
70
        } else {
71
            write_ndigit_number_u(2, top_line, (uint16_t)(((uint32_t)naviData.GroundSpeed * (uint32_t)9) / (uint32_t)250), 3, 0);
72
        }
459 cascade 73
 
761 - 74
        write_ndigit_number_u(7, top_line, rc_signal, 3, 0);
75
        if (rc_signal <= RCLVL_WRN && last_RC_Quality > RCLVL_WRN) {
76
            for (uint8_t x = 0; x < 4; x++)
77
                write_char_att_xy(7 + x, top_line, BLINK);
78
        } else if (rc_signal > RCLVL_WRN && last_RC_Quality <= RCLVL_WRN) {
79
            for (uint8_t x = 0; x < 4; x++)
80
                write_char_att_xy(7 + x, top_line, 0);
81
        }
459 cascade 82
 
83
 
761 - 84
        if (naviData.NCFlags & NC_FLAG_NOSERIALLINK) {
85
            write_char_xy(11, top_line, 0); // clear
86
        } else {
87
            write_char_xy(11, top_line, 0xC6); // PC icon
88
        }
459 cascade 89
 
761 - 90
        write_ndigit_number_u(13, top_line, naviData.CompassHeading, 3, 0);
459 cascade 91
 
761 - 92
        write_ascii_string_pgm(17, top_line, (const char *)(pgm_read_word(&(directions[heading_conv(naviData.CompassHeading)]))));
459 cascade 93
 
761 - 94
        draw_variometer(21, top_line, naviData.Variometer);
459 cascade 95
 
761 - 96
        if (COSD_FLAGS_CONFIG & COSD_FLAG_GPSHEIGHT) {
97
            if (COSD_FLAGS_CONFIG & COSD_FLAG_FEET) {
98
                // feet
99
                write_ndigit_number_s(23, top_line, (int16_t)(naviData.CurrentPosition.Altitude / 1000 - altimeter_offset) * 32 / 10, 4, 0); // GPS
100
            } else {
101
                if (naviData.CurrentPosition.Altitude / 1000 - altimeter_offset > 10 || naviData.CurrentPosition.Altitude / 1000 - altimeter_offset < -10) {
102
                    // above 10m only write full meters
103
                    write_ndigit_number_s(23, top_line, (int16_t)(naviData.CurrentPosition.Altitude / 1000 - altimeter_offset), 4, 0); // GPS
104
                } else {
105
                    // up to 10m write meters.dm
106
                    write_ndigit_number_s_10th(23, top_line, (int16_t)(naviData.CurrentPosition.Altitude / 100 - altimeter_offset * 10), 3, 0); // GPS
107
                }
108
            }
109
        } else {
110
            if (COSD_FLAGS_CONFIG & COSD_FLAG_FEET) {
766 - 111
                write_ndigit_number_s(23, top_line, naviData.Altimeter / 10 * 32 / 20, 4, 0); // BARO
761 - 112
            } else {
766 - 113
                //cite:killagreg "Faktor 20 bis 21 wäre korrekt." (http://forum.mikrokopter.de/topic-post211192.html#post211192)
114
                if (naviData.Altimeter > 200 || naviData.Altimeter < -200) {
761 - 115
                    // above 10m only write full meters
766 - 116
                    write_ndigit_number_s(23, top_line, naviData.Altimeter / 20, 4, 0); // BARO
761 - 117
                } else {
118
                    // up to 10m write meters.dm
766 - 119
                    write_ndigit_number_s_10th(23, top_line, naviData.Altimeter / 2, 3, 0); // BARO
761 - 120
                }
121
            }
122
        }
757 cascade 123
 
459 cascade 124
 
761 - 125
        // seccond line
126
        draw_compass(11, top_line + 1, naviData.CompassHeading);
465 cascade 127
 
761 - 128
        // TODO: verify correctness
129
        uint16_t heading_home = (naviData.HomePositionDeviation.Bearing + 360 - naviData.CompassHeading) % 360;
130
        //write_char_xy(21, top_line + 1, arrowdir[heading_conv(heading_home)]);
131
        // finer resolution, 0xa0 is first character and we add the index 0 <= index < 16
132
        write_char_xy(21, top_line + 1, 0xa0 + heading_fine_conv(heading_home));
728 cascade 133
 
761 - 134
        if (COSD_FLAGS_CONFIG & COSD_FLAG_FEET) {
135
            // feet
136
            write_ndigit_number_u(23, top_line + 1, naviData.HomePositionDeviation.Distance / 10 * 32 / 10, 4, 0);
137
        } else {
138
            write_ndigit_number_u(23, top_line + 1, naviData.HomePositionDeviation.Distance / 10, 4, 0);
139
        }
459 cascade 140
 
761 - 141
        // center
142
        if (naviData.FCFlags & FLAG_MOTOR_RUN) { // should be engines running
143
            if (!(old_MKFlags & FLAG_MOTOR_RUN)) { // motors just started, clear middle
144
                clear();
145
                // remember current heigth for gps offset
146
                altimeter_offset = naviData.CurrentPosition.Altitude / 1000;
147
                // set wasted counter to current offset
148
                if ((COSD_FLAGS_RUNTIME & COSD_FLAG_STROMREC) && !(COSD_FLAGS_MODES & COSD_FLAG_FCCURRENT)) {
149
                    wasted_ampere_offset = ampere_wasted / 10;
150
                } else if (COSD_FLAGS_MODES & COSD_FLAG_FCCURRENT) {
151
                    wasted_ampere_offset = naviData.UsedCapacity;
152
                }
153
                // update flags to paint display again if needed
154
                COSD_FLAGS_RUNTIME &= ~COSD_ICONS_WRITTEN;
155
            }
156
            if (COSD_FLAGS_MODES & COSD_FLAG_ARTHORIZON) { // horizon
157
                uint8_t horizon_bottom = bottom_line - 1;
158
                if ((COSD_FLAGS_RUNTIME & COSD_FLAG_STROMREC) || (COSD_FLAGS_MODES & COSD_FLAG_FCCURRENT)) {
159
                    horizon_bottom--;
160
                }
161
                if (COSD_FLAGS_MODES & COSD_FLAG_AGGRHORIZON) {
162
                    draw_agressiva_artificial_horizon(top_line + 2, horizon_bottom, naviData.AngleNick, naviData.AngleRoll);
163
                } else {
164
                    draw_artificial_horizon(top_line + 2, horizon_bottom, naviData.AngleNick, naviData.AngleRoll);
165
                }
166
            }
167
            // motors are on, assume we were/are flying
168
            COSD_FLAGS_RUNTIME |= COSD_WASFLYING;
169
        } else {
170
            if ((old_MKFlags & FLAG_MOTOR_RUN)) { // motors just stopped
171
                clear(); // clear whole screen in case there is horizon stuff left
172
                // update flags to paint display again if needed
173
                COSD_FLAGS_RUNTIME &= ~COSD_ICONS_WRITTEN;
174
            }
175
            // stats
176
            if ((COSD_FLAGS_RUNTIME & COSD_WASFLYING) && (COSD_FLAGS_MODES & COSD_FLAG_STATS)) {
177
                uint8_t line = 3;
178
                write_ascii_string_pgm(1, line, (const char *)(pgm_read_word(&(stats_item_pointers[0])))); // max Altitude
179
                write_ascii_string_pgm(1, ++line, (const char *)(pgm_read_word(&(stats_item_pointers[1])))); // max Speed
180
                write_ascii_string_pgm(1, ++line, (const char *)(pgm_read_word(&(stats_item_pointers[2])))); // max Distance
459 cascade 181
 
761 - 182
                if (COSD_FLAGS_CONFIG & COSD_FLAG_FEET) {
183
                    write_ndigit_number_s(16, line - 2, max_Altimeter * 32 / 10, 4, 0);
184
                    write_char_xy(20, line - 2, 0x7E); // small feet ft
185
                    write_ndigit_number_u(17, line - 1, (uint16_t)(((uint32_t)max_GroundSpeed * (uint32_t)279) / (uint32_t)12500), 3, 0);
186
                    write_char_xy(20, line - 1, 0x7D); // mp/h
762 - 187
                    write_ndigit_number_u(16, line - 0, max_Distance / 10 * 32 / 10, 4, 0);
761 - 188
                    write_char_xy(20, line - 0, 0x7E); // small feet ft
189
                } else {
190
                    write_ndigit_number_s(16, line - 2, max_Altimeter, 4, 0);
191
                    write_char_xy(20, line - 2, 204); // small meters m
192
                    write_ndigit_number_u(17, line - 1, (uint16_t)(((uint32_t)max_GroundSpeed * (uint32_t)9) / (uint32_t)250), 3, 0);
193
                    write_char_xy(20, line - 1, 203); // km/h
762 - 194
                    write_ndigit_number_u(16, line - 0, max_Distance / 10, 4, 0);
761 - 195
                    write_char_xy(20, line - 0, 204); // small meters m
196
                }
459 cascade 197
 
761 - 198
                write_ascii_string_pgm(1, ++line, (const char *)(pgm_read_word(&(stats_item_pointers[3])))); // min voltage
199
                write_ndigit_number_u_10th(16, line, min_UBat, 3, 0);
200
                write_char_xy(20, line, 0x9E); // small V
201
                if ((COSD_FLAGS_RUNTIME & COSD_FLAG_STROMREC) || (COSD_FLAGS_MODES & COSD_FLAG_FCCURRENT)) {
202
                    write_ascii_string_pgm(1, ++line, (const char *)(pgm_read_word(&(stats_item_pointers[7])))); // ampere
203
                    write_ndigit_number_u_10th(16, line, max_ampere / 10, 3, 0);
204
                    write_char_xy(20, line, 0x9F); // small A
497 cascade 205
 
761 - 206
                    // wasted mampere in this flight (will count up after landing)
207
                    if ((COSD_FLAGS_RUNTIME & COSD_FLAG_STROMREC) && !(COSD_FLAGS_MODES & COSD_FLAG_FCCURRENT)) {
208
                        write_ndigit_number_u(21, line, (ampere_wasted / 10) - wasted_ampere_offset, 5, 0);
209
                    } else if (COSD_FLAGS_MODES & COSD_FLAG_FCCURRENT) {
459 cascade 210
 
761 - 211
                        write_ndigit_number_u(21, line, naviData.UsedCapacity - wasted_ampere_offset, 5, 0);
212
                    }
497 cascade 213
 
761 - 214
                    write_char_xy(26, line, 0xB5); // mah
215
                }
216
                write_ascii_string_pgm(1, ++line, (const char *)(pgm_read_word(&(stats_item_pointers[4])))); // max time
217
                write_time(14, line, max_FlyingTime);
218
                write_char_xy(20, line, 210); // fly clock
219
                write_ascii_string_pgm(1, ++line, (const char *)(pgm_read_word(&(stats_item_pointers[5])))); // longitude
220
                write_gps_pos(14, line, naviData.CurrentPosition.Longitude);
221
                write_ascii_string_pgm(1, ++line, (const char *)(pgm_read_word(&(stats_item_pointers[6])))); // latitude
222
                write_gps_pos(14, line, naviData.CurrentPosition.Latitude);
223
            } else if (COSD_FLAGS_MODES & COSD_FLAG_ARTHORIZON) { // if no stats there is space horizon
224
                uint8_t horizon_bottom = bottom_line - 1;
225
                if ((COSD_FLAGS_RUNTIME & COSD_FLAG_STROMREC) || (COSD_FLAGS_MODES & COSD_FLAG_FCCURRENT)) {
226
                    horizon_bottom--;
227
                }
228
                if (COSD_FLAGS_MODES & COSD_FLAG_AGGRHORIZON) {
229
                    draw_agressiva_artificial_horizon(top_line + 2, horizon_bottom, naviData.AngleNick, naviData.AngleRoll);
230
                } else {
231
                    draw_artificial_horizon(top_line + 2, horizon_bottom, naviData.AngleNick, naviData.AngleRoll);
232
                }
233
            }
234
        }
235
        if (COSD_FLAGS_MODES & COSD_FLAG_BIGVARIO) {
236
            draw_big_variometer(27, 8, naviData.Variometer);
237
        }
459 cascade 238
 
761 - 239
        // pre-bottom line
240
        if ((COSD_FLAGS_RUNTIME & COSD_FLAG_STROMREC) && !(COSD_FLAGS_MODES & COSD_FLAG_FCCURRENT)) {
241
            //write_ndigit_number_s(3, bottom_line - 1, ampere, 4, 0);
242
            write_ndigit_number_u_10th(3, bottom_line - 1, ampere / 10, 3, 0);
243
            write_ndigit_number_s(10, bottom_line - 1, ampere_wasted / 10, 4, 0);
244
            if (COSD_FLAGS_MODES & COSD_FLAG_STROMVOLT) {
245
                write_ndigit_number_u_10th(17, bottom_line - 1, s_volt, 3, 0);
246
            }
247
        } else if (COSD_FLAGS_MODES & COSD_FLAG_FCCURRENT) {
248
            write_ndigit_number_u_10th(3, bottom_line - 1, naviData.Current, 3, 0);
249
            write_ndigit_number_u(10, bottom_line - 1, naviData.UsedCapacity, 4, 0);
250
        }
459 cascade 251
 
761 - 252
        //DEBUGwrite_ndigit_number_u(1, 5, COSD_FLAGS_MODES, 3, 0);
253
 
254
        // bottom line
255
        draw_battery(2, bottom_line, min_voltage, naviData.UBat, max_voltage);
256
        write_ndigit_number_u_10th(3, bottom_line, naviData.UBat, 3, 0);
257
        if (naviData.UBat <= min_voltage && last_UBat > min_voltage) {
258
            for (uint8_t x = 2; x < 8; x++)
259
                write_char_att_xy(x, bottom_line, BLINK);
260
        } else if (naviData.UBat > min_voltage && last_UBat < min_voltage) {
261
            for (uint8_t x = 2; x < 8; x++)
262
                write_char_att_xy(x, bottom_line, 0);
263
        }
264
 
265
        write_time(8, bottom_line, uptime);
266
        write_time(15, bottom_line, naviData.FlyingTime);
267
 
268
        write_ndigit_number_u(24, bottom_line, naviData.SatsInUse, 2, 0);
269
 
270
        if (naviData.NCFlags & NC_FLAG_MANUAL_CONTROL) {
271
            write_char_xy(23, bottom_line, 0xB3); // rc transmitter
272
        } else {
273
            write_char_xy(23, bottom_line, 0); // clear
274
        }
275
 
276
        if (naviData.NCFlags & NC_FLAG_CH) {
277
            write_char_xy(27, bottom_line, 231); // gps ch
278
        } else if (naviData.NCFlags & NC_FLAG_PH) {
279
            write_char_xy(27, bottom_line, 230); // gps ph
280
        } else { // (naviData.NCFlags & NC_FLAG_FREE)
281
            write_char_xy(27, bottom_line, 201); // sat2 (free)
282
        }
283
    }
284
 
285
    //write_number_s(8, 5, RxDataLen);
286
    //write_number_s(16, 5, setsReceived++);
287
 
288
    // remember statistics (only when engines running)
289
    if (naviData.FCFlags & FLAG_MOTOR_RUN) {
290
        if (COSD_FLAGS_CONFIG & COSD_FLAG_GPSHEIGHT) {
291
            if (naviData.CurrentPosition.Altitude / 1000 - altimeter_offset > max_Altimeter) max_Altimeter = naviData.CurrentPosition.Altitude / 1000;
292
        } else {
772 - 293
            if (naviData.Altimeter / 20 > max_Altimeter) max_Altimeter = naviData.Altimeter / 20;
761 - 294
        }
295
        if (naviData.GroundSpeed > max_GroundSpeed) max_GroundSpeed = naviData.GroundSpeed;
296
        if (naviData.HomePositionDeviation.Distance > max_Distance) {
297
            max_Distance = naviData.HomePositionDeviation.Distance;
298
        }
299
        if (naviData.UBat < min_UBat) min_UBat = naviData.UBat;
300
        if (naviData.FlyingTime > max_FlyingTime) max_FlyingTime = naviData.FlyingTime;
301
        if ((COSD_FLAGS_RUNTIME & COSD_FLAG_STROMREC) && !(COSD_FLAGS_MODES & COSD_FLAG_FCCURRENT)) {
302
            if (ampere > max_ampere) max_ampere = ampere;
303
        } else if (COSD_FLAGS_MODES & COSD_FLAG_FCCURRENT) {
304
            if (naviData.Current * 10 > max_ampere) max_ampere = naviData.Current * 10;
305
        }
306
    }
307
 
308
    // remember last values
309
    last_RC_Quality = rc_signal;
310
    last_UBat = naviData.UBat;
311
    old_MKFlags = naviData.FCFlags;
312
    seconds_since_last_data = 0;
313
 
314
    return 0;
459 cascade 315
}
316
 
497 cascade 317
#endif