bye room
This commit is contained in:
		| @@ -1,107 +0,0 @@ | ||||
| <template> | ||||
| <canvas width="224" height="128"></canvas> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import { defineComponent } from 'vue'; | ||||
| import * as THREE from 'three'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default defineComponent({ | ||||
| 	data() { | ||||
| 		return { | ||||
| 			selected: null, | ||||
| 			objectHeight: 0, | ||||
| 			orbitRadius: 5 | ||||
| 		}; | ||||
| 	}, | ||||
|  | ||||
| 	mounted() { | ||||
| 		const canvas = this.$el; | ||||
|  | ||||
| 		const width = canvas.width; | ||||
| 		const height = canvas.height; | ||||
|  | ||||
| 		const scene = new THREE.Scene(); | ||||
|  | ||||
| 		const renderer = new THREE.WebGLRenderer({ | ||||
| 			canvas: canvas, | ||||
| 			antialias: true, | ||||
| 			alpha: false | ||||
| 		}); | ||||
| 		renderer.setPixelRatio(window.devicePixelRatio); | ||||
| 		renderer.setSize(width, height); | ||||
| 		renderer.setClearColor(0x000000); | ||||
| 		renderer.autoClear = false; | ||||
| 		renderer.shadowMap.enabled = true; | ||||
| 		renderer.shadowMap.cullFace = THREE.CullFaceBack; | ||||
|  | ||||
| 		const camera = new THREE.PerspectiveCamera(75, width / height, 0.1, 100); | ||||
| 		camera.zoom = 10; | ||||
| 		camera.position.x = 0; | ||||
| 		camera.position.y = 2; | ||||
| 		camera.position.z = 0; | ||||
| 		camera.updateProjectionMatrix(); | ||||
| 		scene.add(camera); | ||||
|  | ||||
| 		const ambientLight = new THREE.AmbientLight(0xffffff, 1); | ||||
| 		ambientLight.castShadow = false; | ||||
| 		scene.add(ambientLight); | ||||
|  | ||||
| 		const light = new THREE.PointLight(0xffffff, 1, 100); | ||||
| 		light.position.set(3, 3, 3); | ||||
| 		scene.add(light); | ||||
|  | ||||
| 		const grid = new THREE.GridHelper(5, 16, 0x444444, 0x222222); | ||||
| 		scene.add(grid); | ||||
|  | ||||
| 		const render = () => { | ||||
| 			const timer = Date.now() * 0.0004; | ||||
| 			requestAnimationFrame(render); | ||||
| 			 | ||||
| 			camera.position.y = Math.sin(Math.PI / 6) * this.orbitRadius;	// Math.PI / 6 => 30deg | ||||
| 			camera.position.z = Math.cos(timer) * this.orbitRadius; | ||||
| 			camera.position.x = Math.sin(timer) * this.orbitRadius; | ||||
| 			camera.lookAt(new THREE.Vector3(0, this.objectHeight / 2, 0)); | ||||
| 			renderer.render(scene, camera); | ||||
| 		}; | ||||
|  | ||||
| 		this.selected = selected => { | ||||
| 			const obj = selected.clone(); | ||||
|  | ||||
| 			// Remove current object | ||||
| 			const current = scene.getObjectByName('obj'); | ||||
| 			if (current != null) { | ||||
| 				scene.remove(current); | ||||
| 			} | ||||
|  | ||||
| 			// Add new object | ||||
| 			obj.name = 'obj'; | ||||
| 			obj.position.x = 0; | ||||
| 			obj.position.y = 0; | ||||
| 			obj.position.z = 0; | ||||
| 			obj.rotation.x = 0; | ||||
| 			obj.rotation.y = 0; | ||||
| 			obj.rotation.z = 0; | ||||
| 			obj.traverse(child => { | ||||
| 				if (child instanceof THREE.Mesh) { | ||||
| 					child.material = child.material.clone(); | ||||
| 					return child.material.emissive.setHex(0x000000); | ||||
| 				} | ||||
| 			}); | ||||
| 			const objectBoundingBox = new THREE.Box3().setFromObject(obj); | ||||
| 			this.objectHeight = objectBoundingBox.max.y - objectBoundingBox.min.y; | ||||
|  | ||||
| 			const objectWidth = objectBoundingBox.max.x - objectBoundingBox.min.x; | ||||
| 			const objectDepth = objectBoundingBox.max.z - objectBoundingBox.min.z; | ||||
|  | ||||
| 			const horizontal = Math.hypot(objectWidth, objectDepth) / camera.aspect; | ||||
| 			this.orbitRadius = Math.max(horizontal, this.objectHeight) * camera.zoom * 0.625 / Math.tan(camera.fov * 0.5 * (Math.PI / 180)); | ||||
| 		 | ||||
| 			scene.add(obj); | ||||
| 		}; | ||||
|  | ||||
| 		render(); | ||||
| 	}, | ||||
| }); | ||||
| </script> | ||||
| @@ -1,279 +0,0 @@ | ||||
| <template> | ||||
| <div class="hveuntkp"> | ||||
| 	<div v-if="objectSelected" class="controller _section"> | ||||
| 		<div class="_content"> | ||||
| 			<p class="name">{{ selectedFurnitureName }}</p> | ||||
| 			<XPreview ref="preview"/> | ||||
| 			<template v-if="selectedFurnitureInfo.props"> | ||||
| 				<div v-for="k in Object.keys(selectedFurnitureInfo.props)" :key="k"> | ||||
| 					<p>{{ k }}</p> | ||||
| 					<template v-if="selectedFurnitureInfo.props[k] === 'image'"> | ||||
| 						<MkButton @click="chooseImage(k, $event)">{{ $ts._rooms.chooseImage }}</MkButton> | ||||
| 					</template> | ||||
| 					<template v-else-if="selectedFurnitureInfo.props[k] === 'color'"> | ||||
| 						<input type="color" :value="selectedFurnitureProps ? selectedFurnitureProps[k] : null" @change="updateColor(k, $event)"/> | ||||
| 					</template> | ||||
| 				</div> | ||||
| 			</template> | ||||
| 		</div> | ||||
| 		<div class="_content"> | ||||
| 			<MkButton inline :primary="isTranslateMode" @click="translate()"><i class="fas fa-arrows-alt"></i> {{ $ts._rooms.translate }}</MkButton> | ||||
| 			<MkButton inline :primary="isRotateMode" @click="rotate()"><i class="fas fa-undo"></i> {{ $ts._rooms.rotate }}</MkButton> | ||||
| 			<MkButton v-if="isTranslateMode || isRotateMode" inline @click="exit()"><i class="fas fa-ban"></i> {{ $ts._rooms.exit }}</MkButton> | ||||
| 		</div> | ||||
| 		<div class="_content"> | ||||
| 			<MkButton @click="remove()"><i class="fas fa-trash-alt"></i> {{ $ts._rooms.remove }}</MkButton> | ||||
| 		</div> | ||||
| 	</div> | ||||
|  | ||||
| 	<div v-if="isMyRoom" class="menu _section"> | ||||
| 		<div class="_content"> | ||||
| 			<MkButton @click="add()"><i class="fas fa-box-open"></i> {{ $ts._rooms.addFurniture }}</MkButton> | ||||
| 		</div> | ||||
| 		<div class="_content"> | ||||
| 			<MkSelect :model-value="roomType" @update:modelValue="updateRoomType($event)"> | ||||
| 				<template #label>{{ $ts._rooms.roomType }}</template> | ||||
| 				<option value="default">{{ $ts._rooms._roomType.default }}</option> | ||||
| 				<option value="washitsu">{{ $ts._rooms._roomType.washitsu }}</option> | ||||
| 			</MkSelect> | ||||
| 			<label v-if="roomType === 'default'"> | ||||
| 				<span>{{ $ts._rooms.carpetColor }}</span> | ||||
| 				<input type="color" :value="carpetColor" @change="updateCarpetColor($event)"/> | ||||
| 			</label> | ||||
| 		</div> | ||||
| 		<div class="_content"> | ||||
| 			<MkButton inline :disabled="!changed" primary @click="save()"><i class="fas fa-save"></i> {{ $ts.save }}</MkButton> | ||||
| 			<MkButton inline @click="clear()"><i class="fas fa-broom"></i> {{ $ts._rooms.clear }}</MkButton> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import { computed, defineComponent } from 'vue'; | ||||
| import { Room } from '@/scripts/room/room'; | ||||
| import * as Acct from 'misskey-js/built/acct'; | ||||
| import XPreview from './preview.vue'; | ||||
| const storeItems = require('@/scripts/room/furnitures.json5'); | ||||
| import { query as urlQuery } from '@/scripts/url'; | ||||
| import MkButton from '@/components/ui/button.vue'; | ||||
| import MkSelect from '@/components/form/select.vue'; | ||||
| import { selectFile } from '@/scripts/select-file'; | ||||
| import * as os from '@/os'; | ||||
| import { ColdDeviceStorage } from '@/store'; | ||||
| import * as symbols from '@/symbols'; | ||||
|  | ||||
| let room: Room; | ||||
|  | ||||
| export default defineComponent({ | ||||
| 	components: { | ||||
| 		XPreview, | ||||
| 		MkButton, | ||||
| 		MkSelect, | ||||
| 	}, | ||||
|  | ||||
| 	beforeRouteLeave(to, from, next) { | ||||
| 		if (this.changed) { | ||||
| 			os.confirm({ | ||||
| 				type: 'warning', | ||||
| 				text: this.$ts.leaveConfirm, | ||||
| 			}).then(({ canceled }) => { | ||||
| 				if (canceled) { | ||||
| 					next(false); | ||||
| 				} else { | ||||
| 					next(); | ||||
| 				} | ||||
| 			}); | ||||
| 		} else { | ||||
| 			next(); | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	props: { | ||||
| 		acct: { | ||||
| 			type: String, | ||||
| 			required: true | ||||
| 		}, | ||||
| 	}, | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			[symbols.PAGE_INFO]: computed(() => this.user ? { | ||||
| 				title: this.$ts.room, | ||||
| 				avatar: this.user, | ||||
| 			} : null), | ||||
| 			user: null, | ||||
| 			objectSelected: false, | ||||
| 			selectedFurnitureName: null, | ||||
| 			selectedFurnitureInfo: null, | ||||
| 			selectedFurnitureProps: null, | ||||
| 			roomType: null, | ||||
| 			carpetColor: null, | ||||
| 			isTranslateMode: false, | ||||
| 			isRotateMode: false, | ||||
| 			isMyRoom: false, | ||||
| 			changed: false, | ||||
| 		}; | ||||
| 	}, | ||||
|  | ||||
| 	async mounted() { | ||||
| 		window.addEventListener('beforeunload', this.beforeunload); | ||||
|  | ||||
| 		this.user = await os.api('users/show', { | ||||
| 			...Acct.parse(this.acct) | ||||
| 		}); | ||||
|  | ||||
| 		this.isMyRoom = this.$i && (this.$i.id === this.user.id); | ||||
|  | ||||
| 		const roomInfo = await os.api('room/show', { | ||||
| 			userId: this.user.id | ||||
| 		}); | ||||
|  | ||||
| 		this.roomType = roomInfo.roomType; | ||||
| 		this.carpetColor = roomInfo.carpetColor; | ||||
|  | ||||
| 		room = new Room(this.user, this.isMyRoom, roomInfo, this.$el, { | ||||
| 			graphicsQuality: ColdDeviceStorage.get('roomGraphicsQuality'), | ||||
| 			onChangeSelect: obj => { | ||||
| 				this.objectSelected = obj != null; | ||||
| 				if (obj) { | ||||
| 					const f = room.findFurnitureById(obj.name); | ||||
| 					this.selectedFurnitureName = this.$t('_rooms._furnitures.' + f.type); | ||||
| 					this.selectedFurnitureInfo = storeItems.find(x => x.id === f.type); | ||||
| 					this.selectedFurnitureProps = f.props | ||||
| 						? JSON.parse(JSON.stringify(f.props)) // Disable reactivity | ||||
| 						: null; | ||||
| 					this.$nextTick(() => { | ||||
| 						this.$refs.preview.selected(obj); | ||||
| 					}); | ||||
| 				} | ||||
| 			}, | ||||
| 			useOrthographicCamera: ColdDeviceStorage.get('roomUseOrthographicCamera'), | ||||
| 		}); | ||||
| 	}, | ||||
|  | ||||
| 	beforeUnmount() { | ||||
| 		room.destroy(); | ||||
| 		window.removeEventListener('beforeunload', this.beforeunload); | ||||
| 	}, | ||||
|  | ||||
| 	methods: { | ||||
| 		beforeunload(e: BeforeUnloadEvent) { | ||||
| 			if (this.changed) { | ||||
| 				e.preventDefault(); | ||||
| 				e.returnValue = ''; | ||||
| 			} | ||||
| 		}, | ||||
|  | ||||
| 		async add() { | ||||
| 			const { canceled, result: id } = await os.select({ | ||||
| 				title: this.$ts._rooms.addFurniture, | ||||
| 				items: storeItems.map(item => ({ | ||||
| 					value: item.id, text: this.$t('_rooms._furnitures.' + item.id) | ||||
| 				})) | ||||
| 			}); | ||||
| 			if (canceled) return; | ||||
| 			room.addFurniture(id); | ||||
| 			this.changed = true; | ||||
| 		}, | ||||
|  | ||||
| 		remove() { | ||||
| 			this.isTranslateMode = false; | ||||
| 			this.isRotateMode = false; | ||||
| 			room.removeFurniture(); | ||||
| 			this.changed = true; | ||||
| 		}, | ||||
|  | ||||
| 		save() { | ||||
| 			os.api('room/update', { | ||||
| 				room: room.getRoomInfo() | ||||
| 			}).then(() => { | ||||
| 				this.changed = false; | ||||
| 				os.success(); | ||||
| 			}).catch((e: any) => { | ||||
| 				os.alert({ | ||||
| 					type: 'error', | ||||
| 					text: e.message | ||||
| 				}); | ||||
| 			}); | ||||
| 		}, | ||||
|  | ||||
| 		clear() { | ||||
| 			os.confirm({ | ||||
| 				type: 'warning', | ||||
| 				text: this.$ts._rooms.clearConfirm, | ||||
| 			}).then(({ canceled }) => { | ||||
| 				if (canceled) return; | ||||
| 				room.removeAllFurnitures(); | ||||
| 				this.changed = true; | ||||
| 			}); | ||||
| 		}, | ||||
|  | ||||
| 		chooseImage(key, e) { | ||||
| 			selectFile(e.currentTarget || e.target, null).then(file => { | ||||
| 				room.updateProp(key, `/proxy/?${urlQuery({ url: file.thumbnailUrl })}`); | ||||
| 				this.$refs.preview.selected(room.getSelectedObject()); | ||||
| 				this.changed = true; | ||||
| 			}); | ||||
| 		}, | ||||
|  | ||||
| 		updateColor(key, ev) { | ||||
| 			room.updateProp(key, ev.target.value); | ||||
| 			this.$refs.preview.selected(room.getSelectedObject()); | ||||
| 			this.changed = true; | ||||
| 		}, | ||||
|  | ||||
| 		updateCarpetColor(ev) { | ||||
| 			room.updateCarpetColor(ev.target.value); | ||||
| 			this.carpetColor = ev.target.value; | ||||
| 			this.changed = true; | ||||
| 		}, | ||||
|  | ||||
| 		updateRoomType(type) { | ||||
| 			room.changeRoomType(type); | ||||
| 			this.roomType = type; | ||||
| 			this.changed = true; | ||||
| 		}, | ||||
|  | ||||
| 		translate() { | ||||
| 			if (this.isTranslateMode) { | ||||
| 				this.exit(); | ||||
| 			} else { | ||||
| 				this.isRotateMode = false; | ||||
| 				this.isTranslateMode = true; | ||||
| 				room.enterTransformMode('translate'); | ||||
| 			} | ||||
| 			this.changed = true; | ||||
| 		}, | ||||
|  | ||||
| 		rotate() { | ||||
| 			if (this.isRotateMode) { | ||||
| 				this.exit(); | ||||
| 			} else { | ||||
| 				this.isTranslateMode = false; | ||||
| 				this.isRotateMode = true; | ||||
| 				room.enterTransformMode('rotate'); | ||||
| 			} | ||||
| 			this.changed = true; | ||||
| 		}, | ||||
|  | ||||
| 		exit() { | ||||
| 			this.isTranslateMode = false; | ||||
| 			this.isRotateMode = false; | ||||
| 			room.exitTransformMode(); | ||||
| 			this.changed = true; | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .hveuntkp { | ||||
| 	position: relative; | ||||
| 	min-height: 500px; | ||||
|  | ||||
| 	> ::v-deep(canvas) { | ||||
| 		display: block; | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
		Reference in New Issue
	
	Block a user
	 syuilo
					syuilo