5 * th9x - http://code.google.com/p/th9x
6 * er9x - http://code.google.com/p/er9x
7 * gruvin9x - http://code.google.com/p/gruvin9x
9 * License GPLv2: http://www.gnu.org/licenses/gpl-2.0.html
11 * This program is free software; you can redistribute it and/or modify
12 * it under the terms of the GNU General Public License version 2 as
13 * published by the Free Software Foundation.
15 * This program is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
23 enum LogicalSwitchFields
{
32 LS_FIELD_LAST
= LS_FIELD_COUNT
-1
35 #define CSW_1ST_COLUMN (4*FW-3)
36 #define CSW_2ND_COLUMN (8*FW+2+FW/2)
37 #define CSW_3RD_COLUMN (14*FW+1+FW/2)
38 #define CSW_4TH_COLUMN (22*FW)
39 #define CSW_5TH_COLUMN (26*FW+3)
40 #define CSW_6TH_COLUMN (31*FW+1)
42 void putsEdgeDelayParam(coord_t x
, coord_t y
, LogicalSwitchData
* cs
, uint8_t lattr
, uint8_t rattr
)
44 lcdDrawChar(x
-4, y
, '[');
45 lcdDrawNumber(x
, y
, lswTimerValue(cs
->v2
), LEFT
|PREC1
|lattr
);
46 lcdDrawChar(lcdLastRightPos
, y
, ':');
48 lcdDrawText(lcdLastRightPos
+3, y
, "<<", rattr
);
50 lcdDrawText(lcdLastRightPos
+3, y
, "--", rattr
);
52 lcdDrawNumber(lcdLastRightPos
+3, y
, lswTimerValue(cs
->v2
+cs
->v3
), LEFT
|PREC1
|rattr
);
53 lcdDrawChar(lcdLastRightPos
, y
, ']');
56 void onLogicalSwitchesMenu(const char *result
)
58 int8_t sub
= menuVerticalPosition
;
59 LogicalSwitchData
* cs
= lswAddress(sub
);
61 if (result
== STR_COPY
) {
62 clipboard
.type
= CLIPBOARD_TYPE_CUSTOM_SWITCH
;
63 clipboard
.data
.csw
= *cs
;
65 else if (result
== STR_PASTE
) {
66 *cs
= clipboard
.data
.csw
;
67 storageDirty(EE_MODEL
);
69 else if (result
== STR_CLEAR
) {
70 memset(cs
, 0, sizeof(LogicalSwitchData
));
71 storageDirty(EE_MODEL
);
75 void menuModelLogicalSwitches(event_t event
)
77 INCDEC_DECLARE_VARS(EE_MODEL
);
79 MENU(STR_MENULOGICALSWITCHES
, menuTabModel
, MENU_MODEL_LOGICAL_SWITCHES
, MAX_LOGICAL_SWITCHES
, { NAVIGATION_LINE_BY_LINE
|LS_FIELD_LAST
/*repeated...*/ });
82 int sub
= menuVerticalPosition
;
83 horzpos_t horz
= menuHorizontalPosition
;
86 drawColumnHeader(STR_LSW_HEADERS
, horz
);
89 if (horz
<0 && event
==EVT_KEY_LONG(KEY_ENTER
) && !READ_ONLY()) {
91 LogicalSwitchData
* cs
= lswAddress(sub
);
93 POPUP_MENU_ADD_ITEM(STR_COPY
);
94 if (clipboard
.type
== CLIPBOARD_TYPE_CUSTOM_SWITCH
)
95 POPUP_MENU_ADD_ITEM(STR_PASTE
);
96 if (cs
->func
|| cs
->v1
|| cs
->v2
|| cs
->delay
|| cs
->duration
|| cs
->andsw
)
97 POPUP_MENU_ADD_ITEM(STR_CLEAR
);
98 POPUP_MENU_START(onLogicalSwitchesMenu
);
101 for (uint8_t i
=0; i
<NUM_BODY_LINES
; ++i
) {
102 coord_t y
= MENU_HEADER_HEIGHT
+ 1 + i
*FH
;
103 k
= i
+menuVerticalOffset
;
104 LcdFlags attr
= (sub
==k
? ((s_editMode
>0) ? BLINK
|INVERS
: INVERS
) : 0);
105 LcdFlags attr1
= (horz
==1 ? attr
: 0);
106 LcdFlags attr2
= (horz
==2 ? attr
: 0);
107 LogicalSwitchData
* cs
= lswAddress(k
);
110 unsigned int sw
= SWSRC_SW1
+k
;
111 drawSwitch(0, y
, sw
, (getSwitch(sw
) ? BOLD
: 0) | ((sub
==k
&& CURSOR_ON_LINE()) ? INVERS
: 0));
114 lcdDrawTextAtIndex(CSW_1ST_COLUMN
, y
, STR_VCSWFUNC
, cs
->func
, horz
==0 ? attr
: 0);
117 unsigned int cstate
= lswFamily(cs
->func
);
118 int v1_val
=cs
->v1
, v1_min
=0, v1_max
=MIXSRC_LAST_TELEM
;
119 int v2_min
=0, v2_max
=MIXSRC_LAST_TELEM
;
120 int v3_min
=-1, v3_max
=100;
122 if (cstate
== LS_FAMILY_BOOL
|| cstate
== LS_FAMILY_STICKY
) {
123 drawSwitch(CSW_2ND_COLUMN
, y
, cs
->v1
, attr1
);
124 drawSwitch(CSW_3RD_COLUMN
, y
, cs
->v2
, attr2
);
125 v1_min
= SWSRC_FIRST_IN_LOGICAL_SWITCHES
; v1_max
= SWSRC_LAST_IN_LOGICAL_SWITCHES
;
126 v2_min
= SWSRC_FIRST_IN_LOGICAL_SWITCHES
; v2_max
= SWSRC_LAST_IN_LOGICAL_SWITCHES
;
127 INCDEC_SET_FLAG(EE_MODEL
| INCDEC_SWITCH
);
128 INCDEC_ENABLE_CHECK(isSwitchAvailableInLogicalSwitches
);
130 else if (cstate
== LS_FAMILY_EDGE
) {
131 drawSwitch(CSW_2ND_COLUMN
, y
, cs
->v1
, attr1
);
132 putsEdgeDelayParam(CSW_3RD_COLUMN
, y
, cs
, attr2
, horz
==LS_FIELD_V3
? attr
: 0);
133 v1_min
= SWSRC_FIRST_IN_LOGICAL_SWITCHES
; v1_max
= SWSRC_LAST_IN_LOGICAL_SWITCHES
;
134 v2_min
=-129; v2_max
= 122;
135 v3_max
= 222 - cs
->v2
;
137 INCDEC_SET_FLAG(EE_MODEL
| INCDEC_SWITCH
);
138 INCDEC_ENABLE_CHECK(isSwitchAvailableInLogicalSwitches
);
141 INCDEC_SET_FLAG(EE_MODEL
);
142 INCDEC_ENABLE_CHECK(NULL
);
145 else if (cstate
== LS_FAMILY_COMP
) {
147 drawSource(CSW_2ND_COLUMN
, y
, v1_val
, attr1
);
148 drawSource(CSW_3RD_COLUMN
, y
, cs
->v2
, attr2
);
149 INCDEC_SET_FLAG(EE_MODEL
| INCDEC_SOURCE
);
150 INCDEC_ENABLE_CHECK(isSourceAvailable
);
152 else if (cstate
== LS_FAMILY_TIMER
) {
153 lcdDrawNumber(CSW_2ND_COLUMN
, y
, lswTimerValue(cs
->v1
), LEFT
|PREC1
|attr1
);
154 lcdDrawNumber(CSW_3RD_COLUMN
, y
, lswTimerValue(cs
->v2
), LEFT
|PREC1
|attr2
);
155 v1_min
= v2_min
= -128;
156 v1_max
= v2_max
= 122;
157 INCDEC_SET_FLAG(EE_MODEL
);
158 INCDEC_ENABLE_CHECK(NULL
);
162 drawSource(CSW_2ND_COLUMN
, y
, v1_val
, attr1
);
164 INCDEC_SET_FLAG(EE_MODEL
| INCDEC_SOURCE
);
165 INCDEC_ENABLE_CHECK(isSourceAvailableInCustomSwitches
);
168 INCDEC_SET_FLAG(EE_MODEL
);
169 INCDEC_ENABLE_CHECK(NULL
);
171 v2_max
= getMaximumValue(v1_val
);
173 if (v1_val
<= MIXSRC_LAST_CH
) {
174 drawSourceCustomValue(CSW_3RD_COLUMN
, y
, v1_val
, calc100toRESX(cs
->v2
), LEFT
|attr2
);
177 drawSourceCustomValue(CSW_3RD_COLUMN
, y
, v1_val
, cs
->v2
, LEFT
|attr2
);
182 drawSwitch(CSW_4TH_COLUMN
, y
, cs
->andsw
, horz
==LS_FIELD_ANDSW
? attr
: 0);
185 if (cs
->duration
> 0)
186 lcdDrawNumber(CSW_5TH_COLUMN
, y
, cs
->duration
, (horz
==LS_FIELD_DURATION
? attr
: 0)|PREC1
|LEFT
);
188 lcdDrawMMM(CSW_5TH_COLUMN
, y
, horz
==LS_FIELD_DURATION
? attr
: 0);
191 if (cstate
== LS_FAMILY_EDGE
) {
192 lcdDrawText(CSW_6TH_COLUMN
, y
, STR_NA
);
193 if (attr
&& horz
== LS_FIELD_DELAY
) {
194 REPEAT_LAST_CURSOR_MOVE();
197 else if (cs
->delay
> 0) {
198 lcdDrawNumber(CSW_6TH_COLUMN
, y
, cs
->delay
, (horz
==LS_FIELD_DELAY
? attr
: 0)|PREC1
|LEFT
);
201 lcdDrawMMM(CSW_6TH_COLUMN
, y
, horz
==LS_FIELD_DELAY
? attr
: 0);
204 if (attr
&& horz
== LS_FIELD_V3
&& cstate
!= LS_FAMILY_EDGE
) {
205 REPEAT_LAST_CURSOR_MOVE();
208 if (s_editMode
>0 && attr
) {
210 case LS_FIELD_FUNCTION
:
212 cs
->func
= checkIncDec(event
, cs
->func
, 0, LS_FUNC_MAX
, EE_MODEL
, isLogicalSwitchFunctionAvailable
);
213 uint8_t new_cstate
= lswFamily(cs
->func
);
214 if (cstate
!= new_cstate
) {
215 unsigned int save_func
= cs
->func
;
216 memset(cs
, 0, sizeof(LogicalSwitchData
));
217 cs
->func
= save_func
;
218 if (new_cstate
== LS_FAMILY_TIMER
) {
219 cs
->v1
= cs
->v2
= -119;
221 else if (new_cstate
== LS_FAMILY_EDGE
) {
228 cs
->v1
= CHECK_INCDEC_PARAM(event
, v1_val
, v1_min
, v1_max
);
231 if (v1_val
>= MIXSRC_FIRST_TIMER
) {
232 INCDEC_SET_FLAG(EE_MODEL
| INCDEC_REP10
| NO_INCDEC_MARKS
);
234 cs
->v2
= CHECK_INCDEC_PARAM(event
, cs
->v2
, v2_min
, v2_max
);
235 if (cstate
==LS_FAMILY_OFS
&& cs
->v1
!=0 && event
==EVT_KEY_LONG(KEY_ENTER
)) {
237 getvalue_t x
= getValue(v1_val
);
238 if (v1_val
<= MIXSRC_LAST_CH
) {
239 cs
->v2
= calcRESXto100(x
);
241 storageDirty(EE_MODEL
);
245 cs
->v3
= CHECK_INCDEC_PARAM(event
, cs
->v3
, v3_min
, v3_max
);
248 INCDEC_SET_FLAG(EE_MODEL
| INCDEC_SWITCH
);
249 INCDEC_ENABLE_CHECK(isSwitchAvailableInLogicalSwitches
);
250 cs
->andsw
= CHECK_INCDEC_PARAM(event
, cs
->andsw
, -MAX_LS_ANDSW
, MAX_LS_ANDSW
);
252 case LS_FIELD_DURATION
:
253 CHECK_INCDEC_MODELVAR_ZERO(event
, cs
->duration
, MAX_LS_DURATION
);
256 CHECK_INCDEC_MODELVAR_ZERO(event
, cs
->delay
, MAX_LS_DELAY
);