2020-10-06 15:01:36 +00:00
|
|
|
<template>
|
|
|
|
<div class="home h-full overflow-hidden">
|
2021-04-06 21:01:54 +00:00
|
|
|
<ErrorMessage ref="errorMessage" />
|
2020-10-06 15:01:36 +00:00
|
|
|
<div class="flex flex-row h-full">
|
2020-10-29 02:13:32 +00:00
|
|
|
<div class="flex flex-col flex-shrink-0 border-r w-64">
|
|
|
|
<device-select
|
2021-04-06 21:01:54 +00:00
|
|
|
:scan-hint="webUSB"
|
2020-10-29 02:13:32 +00:00
|
|
|
:selected.sync="selectedCalculator"
|
|
|
|
:class="webUSB || 'opacity-50 pointer-events-none'"
|
|
|
|
/>
|
|
|
|
<div class="overflow-auto flex flex-col h-full px-4 py-4">
|
2020-10-06 15:01:36 +00:00
|
|
|
<div v-if="needsDrivers">
|
|
|
|
<h1 class="text-3xl">Drivers required</h1>
|
|
|
|
<p>The WinUSB driver is required to use this device.</p>
|
|
|
|
<p class="text-center mt-2">
|
2020-10-29 02:13:32 +00:00
|
|
|
<a href="#" class="text-blue-600" @click.prevent="installDrivers">
|
|
|
|
See installation instructions
|
|
|
|
</a>
|
2020-10-06 15:01:36 +00:00
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
<div
|
|
|
|
v-else-if="calculator && !calculator.info"
|
2020-10-29 02:13:32 +00:00
|
|
|
class="flex items-center justify-center flex-grow"
|
2020-10-06 15:01:36 +00:00
|
|
|
>
|
|
|
|
<div class="lds-dual-ring" />
|
|
|
|
</div>
|
|
|
|
<div v-else-if="calculator && calculator.info">
|
2020-10-29 02:13:32 +00:00
|
|
|
<calc-info
|
|
|
|
:info="calculator.info"
|
|
|
|
:dev="selectedCalculator"
|
|
|
|
native-upload
|
|
|
|
/>
|
2020-10-06 15:01:36 +00:00
|
|
|
<label class="inline-flex items-center cursor-pointer mr-2 mt-4">
|
|
|
|
<input
|
|
|
|
v-model="showHidden"
|
|
|
|
type="checkbox"
|
|
|
|
class="form-checkbox h-5 w-5 text-blue-600 cursor-pointer"
|
|
|
|
/>
|
|
|
|
<span class="mx-2 text-gray-700 select-none"
|
|
|
|
>Include hidden files</span
|
|
|
|
>
|
|
|
|
</label>
|
|
|
|
</div>
|
2020-10-29 02:13:32 +00:00
|
|
|
<div v-if="!(calculator && !calculator.info)" class="flex-grow" />
|
|
|
|
<div class="mt-4 select-text">
|
|
|
|
<p>
|
|
|
|
<a
|
|
|
|
href="https://lights0123.com/n-link/"
|
|
|
|
target="_blank"
|
|
|
|
class="text-blue-600 underline"
|
|
|
|
>
|
|
|
|
N-Link</a
|
|
|
|
>
|
|
|
|
©
|
|
|
|
<a
|
|
|
|
href="https://lights0123.com/"
|
|
|
|
target="_blank"
|
|
|
|
class="text-blue-600 underline"
|
|
|
|
>
|
|
|
|
Ben Schattinger</a
|
|
|
|
>
|
|
|
|
</p>
|
|
|
|
<p>Licensed under the GPL v3.0</p>
|
|
|
|
</div>
|
2020-10-06 15:01:36 +00:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div class="w-full">
|
2020-10-29 02:13:32 +00:00
|
|
|
<div v-if="webUSB" class="h-full">
|
2020-10-06 15:01:36 +00:00
|
|
|
<file-browser
|
|
|
|
v-if="calculator && calculator.info"
|
|
|
|
:dev="selectedCalculator"
|
|
|
|
:show-hidden="showHidden"
|
|
|
|
native-upload
|
|
|
|
/>
|
|
|
|
</div>
|
2020-10-29 02:13:32 +00:00
|
|
|
<div
|
|
|
|
v-else
|
|
|
|
class="flex flex-col items-center justify-center h-full select-text"
|
|
|
|
>
|
|
|
|
<p class="text-3xl">Your browser doesn't support WebUSB</p>
|
2021-04-06 21:01:54 +00:00
|
|
|
<p class="text-xl">
|
|
|
|
<a
|
|
|
|
href="https://lights0123.com/n-link/"
|
|
|
|
class="text-blue-600 underline inline"
|
|
|
|
>
|
|
|
|
Check out the desktop version instead</a
|
|
|
|
>, or switch to a Chrome-based browser
|
|
|
|
</p>
|
2020-10-29 02:13:32 +00:00
|
|
|
</div>
|
2020-10-06 15:01:36 +00:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script lang="ts">
|
2021-04-06 21:26:08 +00:00
|
|
|
import { Component, Ref, Vue, Watch } from 'vue-property-decorator';
|
2020-10-06 15:01:36 +00:00
|
|
|
import CalcInfo from 'n-link-core/components/CalcInfo.vue';
|
|
|
|
import FileBrowser from 'n-link-core/components/FileBrowser.vue';
|
|
|
|
import DeviceSelect from 'n-link-core/components/DeviceSelect.vue';
|
|
|
|
import '@/components/devices';
|
2021-04-06 21:01:54 +00:00
|
|
|
import ErrorMessage from '@/components/ErrorMessage.vue';
|
2020-10-06 15:01:36 +00:00
|
|
|
|
|
|
|
function sleep(ms: number) {
|
|
|
|
return new Promise((resolve) => setTimeout(resolve, ms));
|
|
|
|
}
|
|
|
|
|
|
|
|
@Component({
|
|
|
|
components: {
|
2021-04-06 21:01:54 +00:00
|
|
|
ErrorMessage,
|
2020-10-06 15:01:36 +00:00
|
|
|
DeviceSelect,
|
|
|
|
FileBrowser,
|
|
|
|
CalcInfo,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
export default class Home extends Vue {
|
|
|
|
selectedCalculator: string | null = null;
|
|
|
|
showHidden = false;
|
2020-10-29 02:19:36 +00:00
|
|
|
webUSB = true;
|
2021-04-06 21:01:54 +00:00
|
|
|
@Ref() readonly errorMessage!: ErrorMessage;
|
2020-10-29 02:19:36 +00:00
|
|
|
|
|
|
|
mounted() {
|
|
|
|
this.webUSB = !!(navigator as any).usb;
|
2021-04-06 21:26:08 +00:00
|
|
|
this.$devices.errorHandler = (e: DOMException) => {
|
|
|
|
this.errorMessage.handleError(e, 'operation');
|
|
|
|
};
|
2020-10-29 02:19:36 +00:00
|
|
|
}
|
2020-10-06 15:01:36 +00:00
|
|
|
|
|
|
|
@Watch('$devices.hasEnumerated')
|
|
|
|
onEnumerated() {
|
|
|
|
const first = Object.keys(this.$devices.devices)[0];
|
|
|
|
if (first) this.selectedCalculator = first;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Watch('$devices.devices')
|
|
|
|
async onDeviceChange() {
|
|
|
|
if (!this.selectedCalculator) {
|
|
|
|
await sleep(1000);
|
|
|
|
if (this.selectedCalculator) return;
|
|
|
|
const first = Object.keys(this.$devices.devices)[0];
|
|
|
|
if (first) this.selectedCalculator = first;
|
|
|
|
} else if (
|
|
|
|
!Object.keys(this.$devices.devices).includes(this.selectedCalculator)
|
|
|
|
) {
|
|
|
|
this.selectedCalculator = null;
|
|
|
|
// go back and choose the first if available
|
|
|
|
this.onDeviceChange();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Watch('selectedCalculator')
|
|
|
|
async onSelectCalculator(dev: string | null) {
|
|
|
|
if (
|
|
|
|
dev &&
|
|
|
|
!this.$devices.devices[dev].info &&
|
|
|
|
!this.$devices.devices[dev].needsDrivers
|
|
|
|
) {
|
|
|
|
try {
|
|
|
|
await this.$devices.open(dev);
|
|
|
|
} catch (e) {
|
2021-04-06 21:01:54 +00:00
|
|
|
console.error({ e });
|
|
|
|
this.errorMessage.handleError(e, 'connection');
|
2020-10-06 15:01:36 +00:00
|
|
|
this.selectedCalculator = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
get calculator() {
|
|
|
|
return (
|
|
|
|
this.selectedCalculator && this.$devices.devices[this.selectedCalculator]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
get needsDrivers() {
|
|
|
|
return (
|
|
|
|
this.selectedCalculator &&
|
|
|
|
this.$devices.devices[this.selectedCalculator]?.needsDrivers
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
installDrivers() {
|
|
|
|
open('https://lights0123.com/n-link/#windows');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<style lang="scss" scoped>
|
|
|
|
.lds-dual-ring {
|
|
|
|
display: inline-block;
|
|
|
|
width: 80px;
|
|
|
|
height: 80px;
|
|
|
|
}
|
|
|
|
|
|
|
|
.lds-dual-ring:after {
|
|
|
|
$color: theme('colors.gray.400');
|
|
|
|
content: ' ';
|
|
|
|
display: block;
|
|
|
|
width: 64px;
|
|
|
|
height: 64px;
|
|
|
|
margin: 8px;
|
|
|
|
border-radius: 50%;
|
|
|
|
border: 6px solid $color;
|
|
|
|
border-color: $color transparent $color transparent;
|
|
|
|
animation: lds-dual-ring 1.2s linear infinite;
|
|
|
|
}
|
|
|
|
|
|
|
|
@keyframes lds-dual-ring {
|
|
|
|
0% {
|
|
|
|
transform: rotate(0deg);
|
|
|
|
}
|
|
|
|
100% {
|
|
|
|
transform: rotate(360deg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
</style>
|