@oicl/openbridge-webcomponents 0.0.15-dev-20240916083108 → 0.0.15-dev-20240923184809
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.storybook/main.ts +8 -0
- package/.storybook/preview.ts +2 -0
- package/__snapshots__/building-blocks-watch--advice.png +0 -0
- package/__snapshots__/building-blocks-watch-flat--primary.png +0 -0
- package/__snapshots__/navigation-instruments-azimuth-thruster--in-command-at-setpoint-disable-auto-setpoint.png +0 -0
- package/__snapshots__/navigation-instruments-azimuth-thruster--in-command-at-setpoint.png +0 -0
- package/__snapshots__/navigation-instruments-azimuth-thruster--in-command.png +0 -0
- package/__snapshots__/navigation-instruments-azimuth-thruster--pod.png +0 -0
- package/__snapshots__/navigation-instruments-azimuth-thruster--single-direction-with-propeller.png +0 -0
- package/__snapshots__/navigation-instruments-azimuth-thruster--single-direction.png +0 -0
- package/__snapshots__/navigation-instruments-azimuth-thruster-labeled--large.png +0 -0
- package/__snapshots__/navigation-instruments-azimuth-thruster-labeled--medium.png +0 -0
- package/__snapshots__/navigation-instruments-azimuth-thruster-labeled--no-command.png +0 -0
- package/__snapshots__/navigation-instruments-compass--primary.png +0 -0
- package/__snapshots__/navigation-instruments-compass-flat--primary.png +0 -0
- package/__snapshots__/navigation-instruments-compass-flat--with-fov-indicator.png +0 -0
- package/__snapshots__/navigation-instruments-main-engine--active.png +0 -0
- package/__snapshots__/navigation-instruments-main-engine--in-command.png +0 -0
- package/__snapshots__/navigation-instruments-main-engine--off.png +0 -0
- package/__snapshots__/navigation-instruments-thruster--tunnel.png +0 -0
- package/custom-elements.json +2184 -660
- package/dist/navigation-instruments/azimuth-thruster/azimuth-thruster.d.ts.map +1 -1
- package/dist/navigation-instruments/azimuth-thruster/azimuth-thruster.js +2 -1
- package/dist/navigation-instruments/azimuth-thruster/azimuth-thruster.js.map +1 -1
- package/dist/navigation-instruments/compass/arrow.d.ts +7 -0
- package/dist/navigation-instruments/compass/arrow.d.ts.map +1 -0
- package/dist/navigation-instruments/compass/arrow.js +59 -0
- package/dist/navigation-instruments/compass/arrow.js.map +1 -0
- package/dist/navigation-instruments/compass/compass.d.ts +23 -0
- package/dist/navigation-instruments/compass/compass.d.ts.map +1 -0
- package/dist/navigation-instruments/compass/compass.js +139 -0
- package/dist/navigation-instruments/compass/compass.js.map +1 -0
- package/dist/navigation-instruments/compass/radial-tickmark.d.ts +4 -0
- package/dist/navigation-instruments/compass/radial-tickmark.d.ts.map +1 -0
- package/dist/navigation-instruments/compass/radial-tickmark.js +69 -0
- package/dist/navigation-instruments/compass/radial-tickmark.js.map +1 -0
- package/dist/navigation-instruments/compass-flat/compass-flat.css.js +29 -0
- package/dist/navigation-instruments/compass-flat/compass-flat.css.js.map +1 -0
- package/dist/navigation-instruments/compass-flat/compass-flat.d.ts +45 -0
- package/dist/navigation-instruments/compass-flat/compass-flat.d.ts.map +1 -0
- package/dist/navigation-instruments/compass-flat/compass-flat.js +223 -0
- package/dist/navigation-instruments/compass-flat/compass-flat.js.map +1 -0
- package/dist/navigation-instruments/main-engine/main-engine.css.js +22 -0
- package/dist/navigation-instruments/main-engine/main-engine.css.js.map +1 -0
- package/dist/navigation-instruments/main-engine/main-engine.d.ts +29 -0
- package/dist/navigation-instruments/main-engine/main-engine.d.ts.map +1 -0
- package/dist/navigation-instruments/main-engine/main-engine.js +196 -0
- package/dist/navigation-instruments/main-engine/main-engine.js.map +1 -0
- package/dist/navigation-instruments/thruster/advice.d.ts.map +1 -1
- package/dist/navigation-instruments/thruster/advice.js +9 -5
- package/dist/navigation-instruments/thruster/advice.js.map +1 -1
- package/dist/navigation-instruments/thruster/thruster.d.ts +54 -1
- package/dist/navigation-instruments/thruster/thruster.d.ts.map +1 -1
- package/dist/navigation-instruments/thruster/thruster.js +163 -99
- package/dist/navigation-instruments/thruster/thruster.js.map +1 -1
- package/dist/navigation-instruments/watch/advice.js +1 -1
- package/dist/navigation-instruments/watch/advice.js.map +1 -1
- package/dist/navigation-instruments/watch/label.d.ts +3 -0
- package/dist/navigation-instruments/watch/label.d.ts.map +1 -0
- package/dist/navigation-instruments/watch/label.js +68 -0
- package/dist/navigation-instruments/watch/label.js.map +1 -0
- package/dist/navigation-instruments/watch/watch.css.js +15 -14
- package/dist/navigation-instruments/watch/watch.css.js.map +1 -1
- package/dist/navigation-instruments/watch/watch.d.ts +3 -0
- package/dist/navigation-instruments/watch/watch.d.ts.map +1 -1
- package/dist/navigation-instruments/watch/watch.js +34 -1
- package/dist/navigation-instruments/watch/watch.js.map +1 -1
- package/dist/navigation-instruments/watch-flat/tickmark-flat.d.ts +20 -0
- package/dist/navigation-instruments/watch-flat/tickmark-flat.d.ts.map +1 -0
- package/dist/navigation-instruments/watch-flat/tickmark-flat.js +53 -0
- package/dist/navigation-instruments/watch-flat/tickmark-flat.js.map +1 -0
- package/dist/navigation-instruments/watch-flat/watch-flat.css.js +32 -0
- package/dist/navigation-instruments/watch-flat/watch-flat.css.js.map +1 -0
- package/dist/navigation-instruments/watch-flat/watch-flat.d.ts +29 -0
- package/dist/navigation-instruments/watch-flat/watch-flat.d.ts.map +1 -0
- package/dist/navigation-instruments/watch-flat/watch-flat.js +184 -0
- package/dist/navigation-instruments/watch-flat/watch-flat.js.map +1 -0
- package/dist/svghelpers/rectangular.d.ts +1 -0
- package/dist/svghelpers/rectangular.d.ts.map +1 -1
- package/dist/svghelpers/rectangular.js +3 -2
- package/dist/svghelpers/rectangular.js.map +1 -1
- package/package.json +16 -11
- package/src/navigation-instruments/azimuth-thruster/azimuth-thruster.ts +1 -0
- package/src/navigation-instruments/compass/arrow.ts +61 -0
- package/src/navigation-instruments/compass/compass.stories.ts +37 -0
- package/src/navigation-instruments/compass/compass.ts +132 -0
- package/src/navigation-instruments/compass/radial-tickmark.ts +77 -0
- package/src/navigation-instruments/compass-flat/compass-flat.css +23 -0
- package/src/navigation-instruments/compass-flat/compass-flat.stories.ts +35 -0
- package/src/navigation-instruments/compass-flat/compass-flat.ts +221 -0
- package/src/navigation-instruments/main-engine/main-engine.css +17 -0
- package/src/navigation-instruments/main-engine/main-engine.stories.ts +54 -0
- package/src/navigation-instruments/main-engine/main-engine.ts +160 -0
- package/src/navigation-instruments/thruster/advice.ts +9 -5
- package/src/navigation-instruments/thruster/thruster.stories.ts +1 -0
- package/src/navigation-instruments/thruster/thruster.ts +205 -113
- package/src/navigation-instruments/watch/advice.ts +1 -1
- package/src/navigation-instruments/watch/label.ts +69 -0
- package/src/navigation-instruments/watch/watch.css +7 -7
- package/src/navigation-instruments/watch/watch.ts +30 -1
- package/src/navigation-instruments/watch-flat/tickmark-flat.ts +62 -0
- package/src/navigation-instruments/watch-flat/watch-flat.css +19 -0
- package/src/navigation-instruments/watch-flat/watch-flat.stories.ts +17 -0
- package/src/navigation-instruments/watch-flat/watch-flat.ts +148 -0
- package/src/palettes/variables.css +1343 -1343
- package/src/svghelpers/rectangular.ts +6 -3
@@ -1,4 +1,4 @@
|
|
1
|
-
import {LitElement, svg, html, css} from 'lit';
|
1
|
+
import {LitElement, svg, html, css, nothing} from 'lit';
|
2
2
|
import {customElement, property} from 'lit/decorators.js';
|
3
3
|
import {InstrumentState} from '../types';
|
4
4
|
import {LinearAdvice, LinearAdviceRaw, renderAdvice} from './advice';
|
@@ -46,6 +46,7 @@ export class ObcThruster extends LitElement {
|
|
46
46
|
singleDirectionHalfSize: this.singleDirectionHalfSize,
|
47
47
|
topPropeller: this.topPropeller,
|
48
48
|
bottomPropeller: this.bottomPropeller,
|
49
|
+
narrow: !this.tunnel,
|
49
50
|
})}
|
50
51
|
</div>`;
|
51
52
|
}
|
@@ -63,11 +64,11 @@ export class ObcThruster extends LitElement {
|
|
63
64
|
`;
|
64
65
|
}
|
65
66
|
|
66
|
-
function thrusterTop(
|
67
|
+
export function thrusterTop(
|
67
68
|
height: number,
|
68
69
|
value: number,
|
69
70
|
colors: {box: string; container: string},
|
70
|
-
hideTicks: boolean
|
71
|
+
options: {hideTicks: boolean; hideContainer: boolean}
|
71
72
|
) {
|
72
73
|
const container = svg`
|
73
74
|
<path transform="translate(0 -2)" d="M -44 0 v -${height - 8} a 8 8 0 0 1 8 -8 h 72 a 8 8 0 0 1 8 8 V 0 Z" fill=${colors.container} stroke="var(--instrument-frame-tertiary-color)" vector-effect="non-scaling-stroke"/>
|
@@ -78,7 +79,7 @@ function thrusterTop(
|
|
78
79
|
|
79
80
|
const nTicks = 2;
|
80
81
|
const delta = height / nTicks;
|
81
|
-
if (!hideTicks) {
|
82
|
+
if (!options.hideTicks) {
|
82
83
|
for (let i = 1; i < nTicks; i++) {
|
83
84
|
tickmarks.push(
|
84
85
|
svg`<line x1="-24" x2="-44" y1=${-i * delta - 2} y2=${
|
@@ -96,54 +97,86 @@ function thrusterTop(
|
|
96
97
|
const barHeight = (height * value) / 100;
|
97
98
|
const barY = -2 - barHeight;
|
98
99
|
const bar = svg`<rect width="40" height=${barHeight} x="-20" y=${barY} fill=${colors.box} stroke=${colors.box} vector-effect="non-scaling-stroke"/>`;
|
99
|
-
|
100
|
-
|
100
|
+
if (options.hideContainer) {
|
101
|
+
return [track, tickmarks, bar];
|
102
|
+
} else {
|
103
|
+
return [container, track, tickmarks, bar];
|
104
|
+
}
|
101
105
|
}
|
102
106
|
|
103
|
-
function thrusterTopSingleSided(
|
107
|
+
export function thrusterTopSingleSided(
|
104
108
|
height: number,
|
105
109
|
value: number,
|
106
110
|
colors: {box: string; container: string},
|
107
|
-
|
108
|
-
|
109
|
-
|
111
|
+
options: {
|
112
|
+
hideTicks: boolean;
|
113
|
+
flipAdicePattern: boolean;
|
114
|
+
hideContainer: boolean;
|
115
|
+
narrow: boolean;
|
116
|
+
},
|
117
|
+
advice: LinearAdviceRaw[]
|
110
118
|
) {
|
111
|
-
const container =
|
119
|
+
const container = options.narrow
|
120
|
+
? svg`
|
112
121
|
<path transform="translate(0 -2)" d="M -32 0 v -${height - 8} a 8 8 0 0 1 8 -8 h 48 a 8 8 0 0 1 8 8 V 0 Z" fill=${colors.container} stroke="var(--instrument-frame-tertiary-color)" vector-effect="non-scaling-stroke"/>
|
122
|
+
`
|
123
|
+
: svg`
|
124
|
+
<path transform="translate(0 -2)" d="M -40 0 v -${height - 8} a 8 8 0 0 1 8 -8 h 56 a 8 8 0 0 1 8 8 V 0 Z" fill=${colors.container} stroke="var(--instrument-frame-tertiary-color)" vector-effect="non-scaling-stroke"/>
|
113
125
|
`;
|
114
|
-
const track =
|
126
|
+
const track = options.narrow
|
127
|
+
? svg`
|
115
128
|
<path transform="translate(0 -2)" d="M -32 0 v -${height - 8} a 8 8 0 0 1 8 -8 h 32 V 0 Z" fill="var(--instrument-frame-secondary-color)" stroke="var(--instrument-frame-tertiary-color)" vector-effect="non-scaling-stroke"/>
|
129
|
+
`
|
130
|
+
: svg`
|
131
|
+
<path transform="translate(0 -2)" d="M -40 0 v -${height - 8} a 8 8 0 0 1 8 -8 h 40 V 0 Z" fill="var(--instrument-frame-secondary-color)" stroke="var(--instrument-frame-tertiary-color)" vector-effect="non-scaling-stroke"/>
|
116
132
|
`;
|
117
133
|
|
118
|
-
const tickmarks = hideTicks
|
134
|
+
const tickmarks = options.hideTicks
|
119
135
|
? []
|
120
136
|
: [singleSidedTickmark(height, 50, TickmarkStyle.hinted)];
|
121
137
|
|
122
138
|
const barHeight = (height * value) / 100;
|
139
|
+
const barWidth = options.narrow ? 40 : 48;
|
140
|
+
const barX = options.narrow ? -32 : -40;
|
123
141
|
const barY = -2 - barHeight;
|
124
|
-
const maskId = flipAdicePattern
|
125
|
-
|
142
|
+
const maskId = options.flipAdicePattern
|
143
|
+
? 'thrusterBarMask1'
|
144
|
+
: 'thrusterBarMask2';
|
145
|
+
// The mask is used to clip the bar to the container shape
|
146
|
+
const mask = options.hideContainer
|
147
|
+
? nothing
|
148
|
+
: svg`
|
126
149
|
<defs>
|
127
150
|
<mask id=${maskId}>
|
128
|
-
<path transform="translate(0 -2)" d="M
|
129
|
-
</defs
|
130
|
-
|
151
|
+
<path transform="translate(0 -2)" d="M ${barX} 0 v -${height - 8} a 8 8 0 0 1 8 -8 h ${barWidth} V 0 Z" fill="white" stroke="white" vector-effect="non-scaling-stroke"/>
|
152
|
+
</defs>`;
|
153
|
+
const maskAttr = options.hideContainer ? '' : `mask="url(#${maskId})"`;
|
154
|
+
const bar = svg`
|
155
|
+
${mask}
|
156
|
+
<rect ${maskAttr} width=${barWidth} height=${barHeight} x=${barX} y=${barY} fill=${colors.box} stroke=${colors.box} vector-effect="non-scaling-stroke"/>`;
|
131
157
|
const advicesSvg = advice.map((a) =>
|
132
|
-
renderAdvice(height, a, flipAdicePattern)
|
158
|
+
renderAdvice(height, a, options.flipAdicePattern)
|
133
159
|
);
|
134
|
-
|
135
|
-
|
160
|
+
const all = [tickmarks, bar, advicesSvg];
|
161
|
+
if (!options.hideContainer) {
|
162
|
+
all.splice(0, 0, [container, track]);
|
163
|
+
}
|
164
|
+
if (!options.narrow) {
|
165
|
+
return svg`<g transform="translate(4 0)">${all}</g>`;
|
166
|
+
} else {
|
167
|
+
return all;
|
168
|
+
}
|
136
169
|
}
|
137
170
|
|
138
|
-
function thrusterBottom(
|
171
|
+
export function thrusterBottom(
|
139
172
|
height: number,
|
140
173
|
value: number,
|
141
174
|
colors: {box: string; container: string},
|
142
|
-
hideTicks: boolean
|
175
|
+
options: {hideTicks: boolean; hideContainer: boolean}
|
143
176
|
) {
|
144
177
|
const container = svg`
|
145
178
|
<g transform="rotate(180)">
|
146
|
-
${thrusterTop(height, value, colors,
|
179
|
+
${thrusterTop(height, value, colors, options)}
|
147
180
|
</g>
|
148
181
|
`;
|
149
182
|
return container;
|
@@ -153,18 +186,23 @@ function thrusterBottomSingleSided(
|
|
153
186
|
height: number,
|
154
187
|
value: number,
|
155
188
|
colors: {box: string; container: string},
|
156
|
-
|
157
|
-
|
189
|
+
options: {
|
190
|
+
hideTicks: boolean;
|
191
|
+
flipAdicePattern: boolean;
|
192
|
+
hideContainer: boolean;
|
193
|
+
narrow: boolean;
|
194
|
+
},
|
195
|
+
advice: LinearAdviceRaw[]
|
158
196
|
) {
|
159
197
|
const container = svg`
|
160
198
|
<g transform="rotate(180) scale(-1,1)">
|
161
|
-
${thrusterTopSingleSided(height, value, colors, hideTicks,
|
199
|
+
${thrusterTopSingleSided(height, value, colors, {hideTicks: options.hideTicks, flipAdicePattern: options.flipAdicePattern, hideContainer: options.hideContainer, narrow: options.narrow}, advice)}
|
162
200
|
</g>
|
163
201
|
`;
|
164
202
|
return container;
|
165
203
|
}
|
166
204
|
|
167
|
-
function setpointSvg(
|
205
|
+
export function setpointSvg(
|
168
206
|
height: number,
|
169
207
|
value: number,
|
170
208
|
setpointAtZero: boolean,
|
@@ -172,12 +210,13 @@ function setpointSvg(
|
|
172
210
|
options: {
|
173
211
|
inCommand: boolean;
|
174
212
|
singleSided: boolean;
|
213
|
+
narrow: boolean;
|
175
214
|
}
|
176
215
|
) {
|
177
216
|
const y = -(setpointAtZero
|
178
217
|
? 0
|
179
218
|
: Math.sign(value) * ((height * Math.abs(value)) / 100 + 2));
|
180
|
-
const extra = options.singleSided ? -12 : 0;
|
219
|
+
const extra = (options.singleSided ? -12 : 0) + (options.narrow ? 0 : 4);
|
181
220
|
let path;
|
182
221
|
if (options.inCommand) {
|
183
222
|
path =
|
@@ -211,6 +250,27 @@ function setpointSvg(
|
|
211
250
|
`;
|
212
251
|
}
|
213
252
|
|
253
|
+
export function atSetpoint(
|
254
|
+
thrust: number,
|
255
|
+
setpoint: number | undefined,
|
256
|
+
options: {
|
257
|
+
autoAtSetpoint: boolean;
|
258
|
+
autoSetpointDeadband: number;
|
259
|
+
touching: boolean;
|
260
|
+
atSetpoint: boolean;
|
261
|
+
}
|
262
|
+
): boolean {
|
263
|
+
if (options.touching) {
|
264
|
+
return false;
|
265
|
+
}
|
266
|
+
|
267
|
+
if (options.autoAtSetpoint && setpoint !== undefined) {
|
268
|
+
return Math.abs(thrust - setpoint) < options.autoSetpointDeadband;
|
269
|
+
}
|
270
|
+
|
271
|
+
return options.atSetpoint;
|
272
|
+
}
|
273
|
+
|
214
274
|
export function thruster(
|
215
275
|
thrust: number,
|
216
276
|
setpoint: number | undefined,
|
@@ -228,92 +288,38 @@ export function thruster(
|
|
228
288
|
advices: LinearAdvice[];
|
229
289
|
topPropeller: PropellerType;
|
230
290
|
bottomPropeller: PropellerType;
|
291
|
+
narrow: boolean;
|
231
292
|
}
|
232
293
|
) {
|
233
|
-
if (
|
234
|
-
|
294
|
+
if (options.tunnel) {
|
295
|
+
thrust = -thrust;
|
296
|
+
setpoint = setpoint === undefined ? undefined : -setpoint;
|
235
297
|
}
|
236
298
|
|
237
|
-
if (options.
|
238
|
-
|
239
|
-
Math.abs(thrust - setpoint) < options.autoSetpointDeadband;
|
299
|
+
if (!options.singleSided && options.advices.length > 0) {
|
300
|
+
throw new Error('Double sided thruster does not support advice');
|
240
301
|
}
|
241
302
|
|
242
|
-
|
243
|
-
options.atSetpoint = false;
|
244
|
-
}
|
303
|
+
options.atSetpoint = atSetpoint(thrust, setpoint, options);
|
245
304
|
|
246
|
-
|
247
|
-
let setPointColor = 'var(--instrument-enhanced-primary-color)';
|
248
|
-
let arrowColor = 'var(--instrument-regular-secondary-color)';
|
249
|
-
let containerBackgroundColor = 'var(--instrument-frame-primary-color)';
|
250
|
-
let zeroLineColor = 'var(--instrument-enhanced-secondary-color)';
|
251
|
-
let hideTicks = false;
|
252
|
-
if (options.atSetpoint) {
|
253
|
-
setPointColor = boxColor;
|
254
|
-
}
|
255
|
-
if (state === InstrumentState.active) {
|
256
|
-
boxColor = 'var(--instrument-regular-secondary-color)';
|
257
|
-
zeroLineColor = 'var(--instrument-regular-secondary-color)';
|
258
|
-
setPointColor = 'var(--instrument-regular-primary-color)';
|
259
|
-
arrowColor = 'var(--instrument-regular-secondary-color)';
|
260
|
-
if (options.atSetpoint) {
|
261
|
-
setPointColor = boxColor;
|
262
|
-
}
|
263
|
-
} else if (state === InstrumentState.loading) {
|
264
|
-
boxColor = 'transparent';
|
265
|
-
setPointColor = 'var(--instrument-frame-tertiary-color)';
|
266
|
-
zeroLineColor = 'var(--instrument-frame-tertiary-color)';
|
267
|
-
arrowColor = 'var(--instrument-regular-secondary-color)';
|
268
|
-
thrust = 0;
|
269
|
-
hideTicks = true;
|
270
|
-
if (setpoint !== undefined) {
|
271
|
-
setpoint = 0;
|
272
|
-
}
|
273
|
-
} else if (state === InstrumentState.off) {
|
274
|
-
boxColor = 'transparent';
|
275
|
-
setPointColor = 'var(--instrument-frame-tertiary-color)';
|
276
|
-
arrowColor = 'var(--instrument-frame-tertiary-color)';
|
277
|
-
zeroLineColor = 'var(--instrument-frame-tertiary-color)';
|
278
|
-
thrust = 0;
|
279
|
-
hideTicks = true;
|
280
|
-
containerBackgroundColor = 'transparent';
|
281
|
-
if (setpoint !== undefined) {
|
282
|
-
setpoint = 0;
|
283
|
-
}
|
284
|
-
}
|
305
|
+
const tc = thrusterColors(options, state);
|
285
306
|
|
286
|
-
|
287
|
-
|
288
|
-
: svg`
|
289
|
-
<rect x="-44" y="-2" width="88" height="4" stroke-width="1" fill=${zeroLineColor} stroke=${zeroLineColor} vector-effect="non-scaling-stroke"/>
|
307
|
+
let centerLine = svg`
|
308
|
+
<rect x="-44" y="-2" width="88" height="4" stroke-width="1" fill=${tc.zeroLineColor} stroke=${tc.zeroLineColor} vector-effect="non-scaling-stroke"/>
|
290
309
|
`;
|
310
|
+
if (options.singleSided) {
|
311
|
+
const width = options.narrow ? 64 : 72;
|
312
|
+
const x = options.narrow ? -32 : -36;
|
313
|
+
centerLine = svg`<rect x=${x} y="-2" width=${width} height="4" stroke-width="1" fill=${tc.zeroLineColor} stroke=${tc.zeroLineColor} vector-effect="non-scaling-stroke"/>`;
|
314
|
+
}
|
291
315
|
|
292
316
|
const setpointAtZero =
|
293
317
|
Math.abs(setpoint || 0) < options.setpointAtZeroDeadband;
|
294
318
|
|
295
|
-
const
|
296
|
-
|
297
|
-
|
298
|
-
|
299
|
-
state = AdviceState.triggered;
|
300
|
-
} else if (a.hinted) {
|
301
|
-
state = AdviceState.hinted;
|
302
|
-
} else {
|
303
|
-
state = AdviceState.regular;
|
304
|
-
}
|
305
|
-
return {
|
306
|
-
min: a.min,
|
307
|
-
max: a.max,
|
308
|
-
type: a.type,
|
309
|
-
state,
|
310
|
-
};
|
311
|
-
});
|
312
|
-
|
313
|
-
const topAdvices = advices.filter((a) => a.min >= 0);
|
314
|
-
const bottomAdvices = advices
|
315
|
-
.filter((a) => a.max <= 0)
|
316
|
-
.map((a) => ({...a, min: -a.max, max: -a.min}));
|
319
|
+
const {topAdvices, bottomAdvices} = convertThrustAdvices(
|
320
|
+
options.advices,
|
321
|
+
thrust
|
322
|
+
);
|
317
323
|
|
318
324
|
const thrusterSvg = [];
|
319
325
|
const baseheight = options.topPropeller === PropellerType.none ? 134 : 106;
|
@@ -323,8 +329,13 @@ export function thruster(
|
|
323
329
|
thrusterTopSingleSided(
|
324
330
|
height,
|
325
331
|
Math.max(thrust, 0),
|
326
|
-
{box: boxColor, container: containerBackgroundColor},
|
327
|
-
|
332
|
+
{box: tc.boxColor, container: tc.containerBackgroundColor},
|
333
|
+
{
|
334
|
+
hideTicks: tc.hideTicks,
|
335
|
+
flipAdicePattern: false,
|
336
|
+
hideContainer: false,
|
337
|
+
narrow: options.narrow,
|
338
|
+
},
|
328
339
|
topAdvices
|
329
340
|
)
|
330
341
|
);
|
@@ -333,8 +344,13 @@ export function thruster(
|
|
333
344
|
thrusterBottomSingleSided(
|
334
345
|
height,
|
335
346
|
Math.max(-thrust, 0),
|
336
|
-
{box: boxColor, container: containerBackgroundColor},
|
337
|
-
|
347
|
+
{box: tc.boxColor, container: tc.containerBackgroundColor},
|
348
|
+
{
|
349
|
+
hideTicks: tc.hideTicks,
|
350
|
+
flipAdicePattern: true,
|
351
|
+
hideContainer: false,
|
352
|
+
narrow: options.narrow,
|
353
|
+
},
|
338
354
|
bottomAdvices
|
339
355
|
)
|
340
356
|
);
|
@@ -345,8 +361,8 @@ export function thruster(
|
|
345
361
|
thrusterTop(
|
346
362
|
height,
|
347
363
|
Math.max(thrust, 0),
|
348
|
-
{box: boxColor, container: containerBackgroundColor},
|
349
|
-
hideTicks
|
364
|
+
{box: tc.boxColor, container: tc.containerBackgroundColor},
|
365
|
+
{hideTicks: tc.hideTicks, hideContainer: false}
|
350
366
|
)
|
351
367
|
);
|
352
368
|
if (!options.singleDirection) {
|
@@ -354,8 +370,8 @@ export function thruster(
|
|
354
370
|
thrusterBottom(
|
355
371
|
height,
|
356
372
|
Math.max(-thrust, 0),
|
357
|
-
{box: boxColor, container: containerBackgroundColor},
|
358
|
-
hideTicks
|
373
|
+
{box: tc.boxColor, container: tc.containerBackgroundColor},
|
374
|
+
{hideTicks: tc.hideTicks, hideContainer: false}
|
359
375
|
)
|
360
376
|
);
|
361
377
|
}
|
@@ -368,12 +384,13 @@ export function thruster(
|
|
368
384
|
setpoint,
|
369
385
|
setpointAtZero,
|
370
386
|
{
|
371
|
-
fill: setPointColor,
|
387
|
+
fill: tc.setPointColor,
|
372
388
|
stroke: 'var(--border-silhouette-color)',
|
373
389
|
},
|
374
390
|
{
|
375
391
|
inCommand: state === InstrumentState.inCommand,
|
376
392
|
singleSided: options.singleSided,
|
393
|
+
narrow: options.narrow,
|
377
394
|
}
|
378
395
|
)
|
379
396
|
);
|
@@ -382,7 +399,7 @@ export function thruster(
|
|
382
399
|
if (options.tunnel) {
|
383
400
|
return svg`
|
384
401
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="-160 -64 320 128" x="-160" y="-64">
|
385
|
-
<g transform="rotate(90)">
|
402
|
+
<g transform="rotate(-90)">
|
386
403
|
${thrusterSvg}
|
387
404
|
</g>
|
388
405
|
</svg>`;
|
@@ -393,7 +410,7 @@ export function thruster(
|
|
393
410
|
viewBox = '-80 -300 160 320';
|
394
411
|
y = -320;
|
395
412
|
}
|
396
|
-
const top = topPropeller(height, arrowColor, options.topPropeller);
|
413
|
+
const top = topPropeller(height, tc.arrowColor, options.topPropeller);
|
397
414
|
const bottom = bottomPropeller(
|
398
415
|
options.singleDirectionHalfSize ? 0.5 : height,
|
399
416
|
options.bottomPropeller
|
@@ -413,3 +430,78 @@ declare global {
|
|
413
430
|
'obc-thruster': ObcThruster;
|
414
431
|
}
|
415
432
|
}
|
433
|
+
|
434
|
+
export function convertThrustAdvices(
|
435
|
+
advices: LinearAdvice[],
|
436
|
+
thrust: number
|
437
|
+
): {topAdvices: LinearAdviceRaw[]; bottomAdvices: LinearAdviceRaw[]} {
|
438
|
+
const rawAdvices: LinearAdviceRaw[] = advices.map((a) => {
|
439
|
+
const triggered = thrust >= a.min && thrust <= a.max;
|
440
|
+
let state: AdviceState;
|
441
|
+
if (triggered) {
|
442
|
+
state = AdviceState.triggered;
|
443
|
+
} else if (a.hinted) {
|
444
|
+
state = AdviceState.hinted;
|
445
|
+
} else {
|
446
|
+
state = AdviceState.regular;
|
447
|
+
}
|
448
|
+
return {
|
449
|
+
min: a.min,
|
450
|
+
max: a.max,
|
451
|
+
type: a.type,
|
452
|
+
state,
|
453
|
+
hinted: a.hinted,
|
454
|
+
};
|
455
|
+
});
|
456
|
+
|
457
|
+
const topAdvices = rawAdvices.filter((a) => a.min >= 0);
|
458
|
+
const bottomAdvices = rawAdvices
|
459
|
+
.filter((a) => a.max <= 0)
|
460
|
+
.map((a) => ({...a, min: -a.max, max: -a.min}));
|
461
|
+
return {topAdvices, bottomAdvices};
|
462
|
+
}
|
463
|
+
|
464
|
+
export function thrusterColors(
|
465
|
+
options: {atSetpoint: boolean; touching: boolean},
|
466
|
+
state: InstrumentState
|
467
|
+
) {
|
468
|
+
let boxColor = 'var(--instrument-enhanced-secondary-color)';
|
469
|
+
let setPointColor = 'var(--instrument-enhanced-primary-color)';
|
470
|
+
let arrowColor = 'var(--instrument-regular-secondary-color)';
|
471
|
+
let containerBackgroundColor = 'var(--instrument-frame-primary-color)';
|
472
|
+
let zeroLineColor = 'var(--instrument-enhanced-secondary-color)';
|
473
|
+
let hideTicks = false;
|
474
|
+
if (options.atSetpoint) {
|
475
|
+
setPointColor = boxColor;
|
476
|
+
}
|
477
|
+
if (state === InstrumentState.active) {
|
478
|
+
boxColor = 'var(--instrument-regular-secondary-color)';
|
479
|
+
zeroLineColor = 'var(--instrument-regular-secondary-color)';
|
480
|
+
setPointColor = 'var(--instrument-regular-primary-color)';
|
481
|
+
arrowColor = 'var(--instrument-regular-secondary-color)';
|
482
|
+
if (options.atSetpoint) {
|
483
|
+
setPointColor = boxColor;
|
484
|
+
}
|
485
|
+
} else if (state === InstrumentState.loading) {
|
486
|
+
boxColor = 'transparent';
|
487
|
+
setPointColor = 'var(--instrument-frame-tertiary-color)';
|
488
|
+
zeroLineColor = 'var(--instrument-frame-tertiary-color)';
|
489
|
+
arrowColor = 'var(--instrument-regular-secondary-color)';
|
490
|
+
hideTicks = true;
|
491
|
+
} else if (state === InstrumentState.off) {
|
492
|
+
boxColor = 'transparent';
|
493
|
+
setPointColor = 'var(--instrument-frame-tertiary-color)';
|
494
|
+
arrowColor = 'var(--instrument-frame-tertiary-color)';
|
495
|
+
zeroLineColor = 'var(--instrument-frame-tertiary-color)';
|
496
|
+
hideTicks = true;
|
497
|
+
containerBackgroundColor = 'transparent';
|
498
|
+
}
|
499
|
+
return {
|
500
|
+
zeroLineColor,
|
501
|
+
boxColor,
|
502
|
+
containerBackgroundColor,
|
503
|
+
hideTicks,
|
504
|
+
setPointColor,
|
505
|
+
arrowColor,
|
506
|
+
};
|
507
|
+
}
|
@@ -112,7 +112,7 @@ export function renderAdvice(advice: AngleAdviceRaw): SVGTemplateResult {
|
|
112
112
|
tickmarkStyle = TickmarkStyle.regular;
|
113
113
|
}
|
114
114
|
return svg`
|
115
|
-
${adviceMask(advice.minAngle, advice.maxAngle, mainColor, mainColor)}
|
115
|
+
${adviceMask(advice.minAngle, advice.maxAngle, advice.state === AdviceState.triggered ? mainColor : 'none', mainColor)}
|
116
116
|
${tickmark(advice.minAngle, TickmarkType.primary, tickmarkStyle, 1)}
|
117
117
|
${tickmark(advice.maxAngle, TickmarkType.primary, tickmarkStyle, 1)}
|
118
118
|
`;
|
@@ -0,0 +1,69 @@
|
|
1
|
+
import {svg, SVGTemplateResult} from 'lit-html';
|
2
|
+
|
3
|
+
export function renderLabels(scale: number): SVGTemplateResult {
|
4
|
+
const labelWidth = 32;
|
5
|
+
const gap = 8;
|
6
|
+
const radius: number = 368 / 2;
|
7
|
+
const labels = [
|
8
|
+
{
|
9
|
+
label: 'E',
|
10
|
+
x: radius + gap / scale + labelWidth / 2,
|
11
|
+
y: 0,
|
12
|
+
class: 'label',
|
13
|
+
},
|
14
|
+
{
|
15
|
+
label: 'S',
|
16
|
+
x: 0,
|
17
|
+
y: radius + gap / scale + labelWidth / 2,
|
18
|
+
class: 'label',
|
19
|
+
},
|
20
|
+
{
|
21
|
+
label: 'W',
|
22
|
+
x: -(radius + gap / scale + labelWidth / 2),
|
23
|
+
y: 0,
|
24
|
+
class: 'label',
|
25
|
+
},
|
26
|
+
];
|
27
|
+
|
28
|
+
let arrow = svg`<defs>
|
29
|
+
<mask id="circleMask">
|
30
|
+
<rect x="-${radius}" y="-${radius}" width="${radius * 2}" height="${radius * 2}" fill="black"/>
|
31
|
+
<circle cx="0" cy="0" r="${radius}" fill="white"/>
|
32
|
+
</mask>
|
33
|
+
</defs>
|
34
|
+
<g mask="url(#circleMask)" transform="translate(0, ${-(radius + 45 / scale)}) scale(${1 / scale}, ${1 / scale})">
|
35
|
+
<path fill-rule="evenodd" clip-rule="evenodd" d="M-32.04 42.7192 0 0 32.039 42.7192C21.627 40.9314 10.922 40 0 40-10.922 40-21.627 40.9314-32.04 42.7192Z"
|
36
|
+
fill="var(--instrument-tick-mark-secondary-color)"/>
|
37
|
+
<path d="M5.003 29H2.091L-3.013 20.264H-3.077C-3.066 20.52-3.056 20.7813-3.045 21.048-3.034 21.3147-3.024 21.5867-3.013 21.864-2.992 22.1307-2.976 22.4027-2.965 22.68-2.954 22.9573-2.944 23.2347-2.933 23.512V29H-4.997V17.576H-2.101L2.987 26.232H3.035C3.024 25.9867 3.014 25.736 3.003 25.48 2.992 25.2133 2.982 24.952 2.971 24.696 2.971 24.4293 2.966 24.1627 2.955 23.896 2.944 23.6293 2.934 23.3627 2.923 23.096V17.576H5.003V29Z" fill="var(--element-active-inverted-color)"/>
|
38
|
+
</g>`;
|
39
|
+
|
40
|
+
if (scale < 0.58) {
|
41
|
+
arrow = svg`
|
42
|
+
<g mask="url(#circleMask)" transform="translate(0, ${-radius})">
|
43
|
+
<path fill-rule="evenodd" clip-rule="evenodd" d="M-17.8457 24.984 0 0 17.8458 24.984C11.9868 24.3338 6.0324 24 0 24-6.0323 24-11.9867 24.3338-17.8457 24.984Z" fill="var(--instrument-frame-tertiary-color)"/>
|
44
|
+
</g>`;
|
45
|
+
|
46
|
+
labels.push({
|
47
|
+
label: 'N',
|
48
|
+
x: 0,
|
49
|
+
y: -(radius + gap / scale + labelWidth / 2),
|
50
|
+
class: 'label',
|
51
|
+
});
|
52
|
+
}
|
53
|
+
|
54
|
+
return svg`
|
55
|
+
${arrow}
|
56
|
+
|
57
|
+
${labels.map(
|
58
|
+
(l) => svg`
|
59
|
+
<text
|
60
|
+
x="${l.x}"
|
61
|
+
y="${l.y}"
|
62
|
+
class="${l.class}"
|
63
|
+
>
|
64
|
+
${l.label}
|
65
|
+
</text>
|
66
|
+
`
|
67
|
+
)}
|
68
|
+
`;
|
69
|
+
}
|
@@ -1,11 +1,11 @@
|
|
1
1
|
* {
|
2
|
-
|
2
|
+
box-sizing: border-box;
|
3
3
|
}
|
4
4
|
|
5
5
|
.label {
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
}
|
6
|
+
@mixin font-body;
|
7
|
+
font-size: calc(16px / var(--scale));
|
8
|
+
fill: var(--element-neutral-color);
|
9
|
+
alignment-baseline: middle;
|
10
|
+
text-anchor: middle;
|
11
|
+
}
|
@@ -14,6 +14,7 @@ import compentStyle from './watch.css?inline';
|
|
14
14
|
import {ResizeController} from '@lit-labs/observers/resize-controller.js';
|
15
15
|
import {AngleAdviceRaw, renderAdvice} from './advice';
|
16
16
|
import {Tickmark, TickmarkStyle, tickmark} from './tickmark';
|
17
|
+
import {renderLabels} from './label';
|
17
18
|
|
18
19
|
@customElement('obc-watch')
|
19
20
|
export class ObcWatch extends LitElement {
|
@@ -27,6 +28,8 @@ export class ObcWatch extends LitElement {
|
|
27
28
|
@property({type: Boolean}) roundInsideCut = false;
|
28
29
|
@property({type: Array, attribute: false}) tickmarks: Tickmark[] = [];
|
29
30
|
@property({type: Array, attribute: false}) advices: AngleAdviceRaw[] = [];
|
31
|
+
@property({type: Boolean}) crosshairEnabled: boolean = false;
|
32
|
+
@property({type: Boolean}) labelFrameEnabled: boolean = false;
|
30
33
|
|
31
34
|
// @ts-expect-error TS6133: The controller unsures that the render
|
32
35
|
// function is called on resize of the element
|
@@ -91,6 +94,29 @@ export class ObcWatch extends LitElement {
|
|
91
94
|
}
|
92
95
|
}
|
93
96
|
|
97
|
+
private renderCrosshair(radius: number): SVGTemplateResult {
|
98
|
+
return svg`
|
99
|
+
<line
|
100
|
+
x1="-${radius}"
|
101
|
+
y1="0"
|
102
|
+
x2="${radius}"
|
103
|
+
y2="0"
|
104
|
+
stroke="var(--instrument-frame-tertiary-color)"
|
105
|
+
stroke-width="1"
|
106
|
+
vector-effect="non-scaling-stroke"
|
107
|
+
/>
|
108
|
+
<line
|
109
|
+
x1="0"
|
110
|
+
y1="-${radius}"
|
111
|
+
x2="0"
|
112
|
+
y2="${radius}"
|
113
|
+
stroke="var(--instrument-frame-tertiary-color)"
|
114
|
+
stroke-width="1"
|
115
|
+
vector-effect="non-scaling-stroke"
|
116
|
+
/>
|
117
|
+
`;
|
118
|
+
}
|
119
|
+
|
94
120
|
override render() {
|
95
121
|
const width = (176 + this.padding) * 2;
|
96
122
|
const viewBox = `-${width / 2} -${width / 2} ${width} ${width}`;
|
@@ -102,6 +128,8 @@ export class ObcWatch extends LitElement {
|
|
102
128
|
const advices = this.advices
|
103
129
|
? this.advices.map((a) => renderAdvice(a))
|
104
130
|
: nothing;
|
131
|
+
const labels = this.labelFrameEnabled ? renderLabels(scale) : nothing;
|
132
|
+
|
105
133
|
return html`
|
106
134
|
<svg
|
107
135
|
width="100%"
|
@@ -109,7 +137,8 @@ export class ObcWatch extends LitElement {
|
|
109
137
|
viewBox=${viewBox}
|
110
138
|
style="--scale: ${scale}"
|
111
139
|
>
|
112
|
-
${this.watchCircle()} ${tickmarks} ${advices} ${angleSetpoint}
|
140
|
+
${this.watchCircle()} ${tickmarks} ${advices} ${angleSetpoint} ${labels}
|
141
|
+
${this.crosshairEnabled ? this.renderCrosshair(320 / 2) : nothing}
|
113
142
|
</svg>
|
114
143
|
`;
|
115
144
|
}
|