Compare commits
	
		
			13 Commits
		
	
	
		
			v0.0.33
			...
			ed4f1d6f2c
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| ed4f1d6f2c | |||
| 73f4a3407f | |||
| 6f11318e84 | |||
| e88ee91f0e | |||
| 3f8daf257c | |||
| dc387acadc | |||
| 68aa41ab96 | |||
| 85b23437b3 | |||
| c59fba817d | |||
| c3415ab75c | |||
| f1d0151d71 | |||
| 3c5c1756d1 | |||
| 6a6b65d1b3 | 
							
								
								
									
										11
									
								
								GNUmakefile
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								GNUmakefile
									
									
									
									
									
								
							| @@ -16,9 +16,9 @@ MAKEFLAGS += --no-builtin-rules | ||||
| ## LD := Linker. | ||||
| ## ANDROID_SDK := Path to the Android SDK. | ||||
|  | ||||
| VERSION_CODE := 40 | ||||
| VERSION_CODE_IOS := 15 | ||||
| VERSION_NUMBER := 0.0.33 | ||||
| VERSION_CODE := 41 | ||||
| VERSION_CODE_IOS := 16 | ||||
| VERSION_NUMBER := 0.2025.8-wip | ||||
| VERSION_NAME := This program kills fascists. | ||||
|  | ||||
| IPHONEOS_VERSION_MIN=14.0 | ||||
| @@ -1140,6 +1140,11 @@ releaseapkgo: out/TildeFriends-arm-release.apk ## Build, install, and run a rele | ||||
| 	@adb shell am start com.unprompted.tildefriends/.TildeFriendsActivity | ||||
| .PHONY: releaseapkgo | ||||
|  | ||||
| x86releaseapkgo: out/TildeFriends-x86-release.apk ## Build, install, and run an x86 release Android APK. | ||||
| 	@adb install -r $< | ||||
| 	@adb shell am start com.unprompted.tildefriends/.TildeFriendsActivity | ||||
| .PHONY: x86releaseapkgo | ||||
|  | ||||
| apklog: ## Display Android log output. | ||||
| 	@adb logcat *:S tildefriends | ||||
| .PHONY: apklog | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| { | ||||
| 	"type": "tildefriends-app", | ||||
| 	"emoji": "🦀", | ||||
| 	"previous": "&DGtlnm5wWRZCgJMF8JsP6VtzNRrd4KLoERJRpFULqOY=.sha256" | ||||
| 	"previous": "&TTGzyovmfKozjELCGPBFLLEXQcpfaArOMmqzemvz9J8=.sha256" | ||||
| } | ||||
|   | ||||
| @@ -25,6 +25,7 @@ class TfElement extends LitElement { | ||||
| 			recent_reactions: {type: Array}, | ||||
| 			is_administrator: {type: Boolean}, | ||||
| 			stay_connected: {type: Boolean}, | ||||
| 			progress: {type: Number}, | ||||
| 		}; | ||||
| 	} | ||||
|  | ||||
| @@ -56,6 +57,7 @@ class TfElement extends LitElement { | ||||
| 		tfrpc.rpc.getHash().then((hash) => self.set_hash(hash)); | ||||
| 		tfrpc.register(function hashChanged(hash) { | ||||
| 			self.set_hash(hash); | ||||
| 			self.reset_progress(); | ||||
| 		}); | ||||
| 		tfrpc.register(async function notifyNewMessage(id) { | ||||
| 			await self.fetch_new_message(id); | ||||
| @@ -450,7 +452,28 @@ class TfElement extends LitElement { | ||||
| 		this.schedule_load_latest(); | ||||
| 	} | ||||
|  | ||||
| 	reset_progress() { | ||||
| 		if (this.progress === undefined) { | ||||
| 			this._progress_start = new Date(); | ||||
| 			requestAnimationFrame(this.update_progress.bind(this)); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	update_progress() { | ||||
| 		if ( | ||||
| 			!this.loading_latest && | ||||
| 			!this.loading_latest_scheduled && | ||||
| 			!this.shadowRoot.getElementById('tf-tab-news')?.is_loading() | ||||
| 		) { | ||||
| 			this.progress = undefined; | ||||
| 			return; | ||||
| 		} | ||||
| 		this.progress = (new Date() - this._progress_start).valueOf(); | ||||
| 		requestAnimationFrame(this.update_progress.bind(this)); | ||||
| 	} | ||||
|  | ||||
| 	schedule_load_latest() { | ||||
| 		this.reset_progress(); | ||||
| 		if (!this.loading_latest) { | ||||
| 			this.shadowRoot.getElementById('tf-tab-news')?.load_latest(); | ||||
| 			this.load(); | ||||
| @@ -495,6 +518,7 @@ class TfElement extends LitElement { | ||||
|  | ||||
| 	async load() { | ||||
| 		this.loading_latest = true; | ||||
| 		this.reset_progress(); | ||||
| 		try { | ||||
| 			let start_time = new Date(); | ||||
| 			let whoami = this.whoami; | ||||
| @@ -603,6 +627,7 @@ class TfElement extends LitElement { | ||||
| 					@channelsetunread=${this.channel_set_unread} | ||||
| 					@refresh=${this.refresh} | ||||
| 					@toggle_stay_connected=${this.toggle_stay_connected} | ||||
| 					@loadmessages=${this.reset_progress} | ||||
| 					.connections=${this.connections} | ||||
| 					.private_messages=${this.private_messages} | ||||
| 					.recent_reactions=${this.recent_reactions} | ||||
| @@ -646,6 +671,7 @@ class TfElement extends LitElement { | ||||
| 	async set_tab(tab) { | ||||
| 		this.tab = tab; | ||||
| 		if (tab === 'news') { | ||||
| 			this.schedule_load_latest(); | ||||
| 			await tfrpc.rpc.setHash('#'); | ||||
| 		} else if (tab === 'connections') { | ||||
| 			await tfrpc.rpc.setHash('#connections'); | ||||
| @@ -751,11 +777,23 @@ class TfElement extends LitElement { | ||||
| 						Loading... | ||||
| 					</div>` | ||||
| 				: this.render_tab(); | ||||
| 		let progress = | ||||
| 			this.progress !== undefined | ||||
| 				? html` | ||||
| 						<div style="position: absolute; width: 100%" id="progress"> | ||||
| 							<div | ||||
| 								class="w3-theme-l3" | ||||
| 								style=${`height: 4px; position: absolute; right: ${Math.cos(this.progress / 250) > 0 ? 'auto' : '0'}; width: ${50 * Math.sin(this.progress / 250) + 50}%`} | ||||
| 							></div> | ||||
| 						</div> | ||||
| 					` | ||||
| 				: undefined; | ||||
| 		return html` | ||||
| 			<div | ||||
| 				style="width: 100vw; min-height: 100vh; height: 100vh; display: flex; flex-direction: column" | ||||
| 				class="w3-theme-dark" | ||||
| 			> | ||||
| 				${progress} | ||||
| 				<div style="flex: 0 0">${tabs}</div> | ||||
| 				<div style="flex: 1 1; overflow: auto; contain: layout"> | ||||
| 					${contents} | ||||
|   | ||||
| @@ -789,60 +789,45 @@ class TfMessageElement extends LitElement { | ||||
| 					</div> | ||||
| 				`); | ||||
| 			} else if (content.type == 'contact') { | ||||
| 				return this.render_frame(html` | ||||
| 					<div class="w3-bar"> | ||||
| 						<div class="w3-bar-item"> | ||||
| 							<tf-user id=${this.message.author} .users=${this.users}></tf-user> | ||||
| 							is | ||||
| 							${content.blocking === true | ||||
| 								? 'blocking' | ||||
| 								: content.blocking === false | ||||
| 									? 'no longer blocking' | ||||
| 									: content.following === true | ||||
| 										? 'following' | ||||
| 										: content.following === false | ||||
| 											? 'no longer following' | ||||
| 											: '?'} | ||||
| 							<tf-user | ||||
| 								id=${this.message.content.contact} | ||||
| 								.users=${this.users} | ||||
| 							></tf-user> | ||||
| 						</div> | ||||
| 						<div class="w3-bar-item w3-right"> | ||||
| 							<button class="w3-button w3-theme-d1" @click=${this.toggle_menu}> | ||||
| 								% | ||||
| 							</button> | ||||
| 							<div | ||||
| 								class="w3-dropdown-content w3-bar-block w3-card-4 w3-theme-l1" | ||||
| 								style="right: 48px" | ||||
| 							> | ||||
| 								<a | ||||
| 									target="_top" | ||||
| 									class="w3-button w3-bar-item" | ||||
| 									href=${'#' + encodeURIComponent(this.message?.id)} | ||||
| 									>View Message</a | ||||
| 								> | ||||
| 								<button | ||||
| 									class="w3-button w3-bar-item w3-border-bottom" | ||||
| 									@click=${this.copy_id} | ||||
| 								> | ||||
| 									Copy ID | ||||
| 								</button> | ||||
| 								${this.drafts[this.message?.id] === undefined | ||||
| 									? html` | ||||
| 											<button | ||||
| 												class="w3-button w3-bar-item" | ||||
| 												@click=${this.show_reply} | ||||
| 											> | ||||
| 												↩️ Reply | ||||
| 											</button> | ||||
| 										` | ||||
| 									: undefined} | ||||
| 				switch (this.format) { | ||||
| 					case 'message': | ||||
| 					default: | ||||
| 						return this.render_frame(html` | ||||
| 							<div class="w3-bar"> | ||||
| 								<div class="w3-bar-item"> | ||||
| 									<tf-user | ||||
| 										id=${this.message.author} | ||||
| 										.users=${this.users} | ||||
| 									></tf-user> | ||||
| 									is | ||||
| 									${content.blocking === true | ||||
| 										? 'blocking' | ||||
| 										: content.blocking === false | ||||
| 											? 'no longer blocking' | ||||
| 											: content.following === true | ||||
| 												? 'following' | ||||
| 												: content.following === false | ||||
| 													? 'no longer following' | ||||
| 													: '?'} | ||||
| 									<tf-user | ||||
| 										id=${this.message.content.contact} | ||||
| 										.users=${this.users} | ||||
| 									></tf-user> | ||||
| 								</div> | ||||
| 								${this.render_menu()} ${this.render_votes()} | ||||
| 								${this.render_actions()} | ||||
| 							</div> | ||||
| 						`); | ||||
| 						break; | ||||
| 					case 'raw': | ||||
| 						return this.render_frame(html` | ||||
| 							${this.render_header()} | ||||
| 							<div class="w3-container">${this.render_raw()}</div> | ||||
| 							${this.render_votes()} ${this.render_actions()} | ||||
| 						</div> | ||||
| 						${this.render_votes()} ${this.render_actions()} | ||||
| 					</div> | ||||
| 				`); | ||||
| 						`); | ||||
| 						break; | ||||
| 				} | ||||
| 			} else if (content.type == 'post') { | ||||
| 				let self = this; | ||||
| 				let body; | ||||
|   | ||||
| @@ -106,6 +106,12 @@ class TfTabNewsFeedElement extends LitElement { | ||||
| 	} | ||||
|  | ||||
| 	async fetch_messages(start_time, end_time) { | ||||
| 		this.dispatchEvent( | ||||
| 			new CustomEvent('loadmessages', { | ||||
| 				bubbles: true, | ||||
| 				composed: true, | ||||
| 			}) | ||||
| 		); | ||||
| 		this.time_loading = [start_time, end_time]; | ||||
| 		let result; | ||||
| 		const k_max_results = 64; | ||||
| @@ -378,7 +384,8 @@ class TfTabNewsFeedElement extends LitElement { | ||||
| 				this.messages = []; | ||||
| 				this._messages_hash = this.hash; | ||||
| 			} | ||||
| 			this._messages_following = this.following; | ||||
| 			this._messages_following = JSON.stringify(this.following); | ||||
| 			this._private_messages = JSON.stringify(this.private_messages); | ||||
| 			let now = new Date().valueOf(); | ||||
| 			let start_time = now - 24 * 60 * 60 * 1000; | ||||
| 			this.start_time = start_time; | ||||
| @@ -421,8 +428,8 @@ class TfTabNewsFeedElement extends LitElement { | ||||
| 		if ( | ||||
| 			!this.messages || | ||||
| 			this._messages_hash !== this.hash || | ||||
| 			JSON.stringify(this._messages_following) !== | ||||
| 				JSON.stringify(this.following) | ||||
| 			this._messages_following !== JSON.stringify(this.following) || | ||||
| 			this._private_messages !== JSON.stringify(this.private_messages) | ||||
| 		) { | ||||
| 			console.log( | ||||
| 				`loading messages for ${this.whoami} (following ${this.following.length})` | ||||
|   | ||||
| @@ -180,6 +180,10 @@ class TfTabNewsElement extends LitElement { | ||||
| 		await this.check_peer_exchange(); | ||||
| 	} | ||||
|  | ||||
| 	is_loading() { | ||||
| 		return this.shadowRoot?.getElementById('news')?.loading; | ||||
| 	} | ||||
|  | ||||
| 	render_sidebar() { | ||||
| 		return html` | ||||
| 			<div | ||||
|   | ||||
							
								
								
									
										37
									
								
								core/app.js
									
									
									
									
									
								
							
							
						
						
									
										37
									
								
								core/app.js
									
									
									
									
									
								
							| @@ -1,7 +1,23 @@ | ||||
| /** | ||||
|  * \file | ||||
|  * \defgroup tfapp Tilde Friends App JS | ||||
|  * Tilde Friends server-side app wrapper. | ||||
|  * @{ | ||||
|  */ | ||||
|  | ||||
| /** \cond */ | ||||
| import * as core from './core.js'; | ||||
|  | ||||
| let gSessionIndex = 0; | ||||
| export {App}; | ||||
| /** \endcond */ | ||||
|  | ||||
| /** A sequence number of apps. */ | ||||
| let g_session_index = 0; | ||||
|  | ||||
| /** | ||||
|  ** App constructor. | ||||
|  ** @return An app instance. | ||||
|  */ | ||||
| function App() { | ||||
| 	this._send_queue = []; | ||||
| 	this.calls = {}; | ||||
| @@ -9,6 +25,12 @@ function App() { | ||||
| 	return this; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  ** Create a function wrapper that when called invokes a function on the app | ||||
|  ** itself. | ||||
|  ** @param api The function and argument names. | ||||
|  ** @return A function. | ||||
|  */ | ||||
| App.prototype.makeFunction = function (api) { | ||||
| 	let self = this; | ||||
| 	let result = function () { | ||||
| @@ -32,6 +54,10 @@ App.prototype.makeFunction = function (api) { | ||||
| 	return result; | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  ** Send a message to the app. | ||||
|  ** @param message The message to send. | ||||
|  */ | ||||
| App.prototype.send = function (message) { | ||||
| 	if (this._send_queue) { | ||||
| 		if (this._on_output) { | ||||
| @@ -46,6 +72,11 @@ App.prototype.send = function (message) { | ||||
| 	} | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  ** App socket handler. | ||||
|  ** @param request The HTTP request of the WebSocket connection. | ||||
|  ** @param response The HTTP response. | ||||
|  */ | ||||
| exports.app_socket = async function socket(request, response) { | ||||
| 	let process; | ||||
| 	let options = {}; | ||||
| @@ -133,7 +164,7 @@ exports.app_socket = async function socket(request, response) { | ||||
| 				options.packageOwner = packageOwner; | ||||
| 				options.packageName = packageName; | ||||
| 				options.url = message.url; | ||||
| 				let sessionId = 'session_' + (gSessionIndex++).toString(); | ||||
| 				let sessionId = 'session_' + (g_session_index++).toString(); | ||||
| 				if (blobId) { | ||||
| 					if (message.edit_only) { | ||||
| 						response.send( | ||||
| @@ -218,4 +249,4 @@ exports.app_socket = async function socket(request, response) { | ||||
| 	response.upgrade(100, {}); | ||||
| }; | ||||
|  | ||||
| export {App}; | ||||
| /** @} */ | ||||
|   | ||||
							
								
								
									
										125
									
								
								core/client.js
									
									
									
									
									
								
							
							
						
						
									
										125
									
								
								core/client.js
									
									
									
									
									
								
							| @@ -72,7 +72,7 @@ class TfNavigationElement extends LitElement { | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * TODOC | ||||
| 	 * Toggle editor visibility. | ||||
| 	 * @param event The HTML event. | ||||
| 	 */ | ||||
| 	toggle_edit(event) { | ||||
| @@ -85,7 +85,7 @@ class TfNavigationElement extends LitElement { | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * TODOC | ||||
| 	 * Remove a stored permission. | ||||
| 	 * @param key The permission to reset. | ||||
| 	 */ | ||||
| 	reset_permission(key) { | ||||
| @@ -93,7 +93,7 @@ class TfNavigationElement extends LitElement { | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * TODOC | ||||
| 	 * Get or create a spark line. | ||||
| 	 * @param key The spark line identifier. | ||||
| 	 * @param options Spark line options. | ||||
| 	 * @return A spark line HTML element. | ||||
| @@ -262,8 +262,8 @@ class TfNavigationElement extends LitElement { | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * TODOC | ||||
| 	 * @returns | ||||
| 	 * Render the permissions popup. | ||||
| 	 * @return Lit HTML. | ||||
| 	 */ | ||||
| 	render_permissions() { | ||||
| 		if (this.show_permissions) { | ||||
| @@ -312,8 +312,8 @@ class TfNavigationElement extends LitElement { | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * TODOC | ||||
| 	 * @returns | ||||
| 	 * Render the navigation bar. | ||||
| 	 * @return Lit HTML. | ||||
| 	 */ | ||||
| 	render() { | ||||
| 		let self = this; | ||||
| @@ -441,7 +441,7 @@ class TfNavigationElement extends LitElement { | ||||
| customElements.define('tf-navigation', TfNavigationElement); | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * A file in the files sidebar. | ||||
|  */ | ||||
| class TfFilesElement extends LitElement { | ||||
| 	/** | ||||
| @@ -467,7 +467,7 @@ class TfFilesElement extends LitElement { | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * TODOC | ||||
| 	 * Select a clicked file. | ||||
| 	 * @param file The file. | ||||
| 	 */ | ||||
| 	file_click(file) { | ||||
| @@ -483,9 +483,9 @@ class TfFilesElement extends LitElement { | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * TODOC | ||||
| 	 * Render a single file in the file list. | ||||
| 	 * @param file The file. | ||||
| 	 * @returns Lit HTML. | ||||
| 	 * @return Lit HTML. | ||||
| 	 */ | ||||
| 	render_file(file) { | ||||
| 		let classes = ['file']; | ||||
| @@ -507,7 +507,7 @@ class TfFilesElement extends LitElement { | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * TODOC | ||||
| 	 * Create a file entry for a dropped file. | ||||
| 	 * @param event The event. | ||||
| 	 */ | ||||
| 	async drop(event) { | ||||
| @@ -533,7 +533,7 @@ class TfFilesElement extends LitElement { | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * TODOC | ||||
| 	 * Called when a file starts being dragged over the file. | ||||
| 	 * @param event The event. | ||||
| 	 */ | ||||
| 	drag_enter(event) { | ||||
| @@ -543,7 +543,7 @@ class TfFilesElement extends LitElement { | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * TODOC | ||||
| 	 * Called when a file stops being dragged over the file. | ||||
| 	 * @param event The event. | ||||
| 	 */ | ||||
| 	drag_leave(event) { | ||||
| @@ -554,7 +554,7 @@ class TfFilesElement extends LitElement { | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * Drag over event. | ||||
| 	 * Called when a file is being dragged over the file. | ||||
| 	 * @param event The event. | ||||
| 	 */ | ||||
| 	drag_over(event) { | ||||
| @@ -562,8 +562,8 @@ class TfFilesElement extends LitElement { | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * TODOC | ||||
| 	 * @returns | ||||
| 	 * Render the file. | ||||
| 	 * @return Lit HTML. | ||||
| 	 */ | ||||
| 	render() { | ||||
| 		let self = this; | ||||
| @@ -610,7 +610,7 @@ class TfFilesElement extends LitElement { | ||||
| customElements.define('tf-files', TfFilesElement); | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * The files pane element. | ||||
|  */ | ||||
| class TfFilesPaneElement extends LitElement { | ||||
| 	/** | ||||
| @@ -635,7 +635,7 @@ class TfFilesPaneElement extends LitElement { | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * TODOC | ||||
| 	 * Set whether the files pane is expanded. | ||||
| 	 * @param expanded Whether the files pane is expanded. | ||||
| 	 */ | ||||
| 	set_expanded(expanded) { | ||||
| @@ -644,8 +644,8 @@ class TfFilesPaneElement extends LitElement { | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * TODOC | ||||
| 	 * @returns | ||||
| 	 * Render the files pane element. | ||||
| 	 * @return Lit HTML. | ||||
| 	 */ | ||||
| 	render() { | ||||
| 		let self = this; | ||||
| @@ -704,7 +704,7 @@ class TfFilesPaneElement extends LitElement { | ||||
| customElements.define('tf-files-pane', TfFilesPaneElement); | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * A tiny graph. | ||||
|  */ | ||||
| class TfSparkLineElement extends LitElement { | ||||
| 	static get properties() { | ||||
| @@ -724,9 +724,9 @@ class TfSparkLineElement extends LitElement { | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * TODOC | ||||
| 	 * @param {*} key | ||||
| 	 * @param {*} value | ||||
| 	 * Add a data point to the graph. | ||||
| 	 * @param key The line to which the point applies. | ||||
| 	 * @param value The numeric value of the data point. | ||||
| 	 */ | ||||
| 	append(key, value) { | ||||
| 		let line = null; | ||||
| @@ -753,9 +753,9 @@ class TfSparkLineElement extends LitElement { | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * TODOC | ||||
| 	 * @param {*} line | ||||
| 	 * @returns | ||||
| 	 * Render a single series line. | ||||
| 	 * @param line The line data. | ||||
| 	 * @return Lit HTML. | ||||
| 	 */ | ||||
| 	render_line(line) { | ||||
| 		if (line?.values?.length >= 2) { | ||||
| @@ -771,8 +771,8 @@ class TfSparkLineElement extends LitElement { | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * TODOC | ||||
| 	 * @returns | ||||
| 	 * Render the graph. | ||||
| 	 * @return Lit HTML. | ||||
| 	 */ | ||||
| 	render() { | ||||
| 		let max = | ||||
| @@ -799,7 +799,9 @@ class TfSparkLineElement extends LitElement { | ||||
|  | ||||
| customElements.define('tf-sparkline', TfSparkLineElement); | ||||
|  | ||||
| // TODOC | ||||
| /** | ||||
|  *  A keyboard key is pressed down. | ||||
|  */ | ||||
| window.addEventListener('keydown', function (event) { | ||||
| 	if (event.keyCode == 83 && (event.altKey || event.ctrlKey)) { | ||||
| 		if (editing()) { | ||||
| @@ -860,24 +862,23 @@ function ensureLoaded(nodes, callback) { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * @returns | ||||
|  * Check whether the editior is currently visible. | ||||
|  * @return true if the editor is visible. | ||||
|  */ | ||||
| function editing() { | ||||
| 	return document.getElementById('editPane').style.display != 'none'; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * @returns | ||||
|  * Check whether only the editor is visible and the app is hidden. | ||||
|  * @return true if the editor is visible and the app is not. | ||||
|  */ | ||||
| function is_edit_only() { | ||||
| 	return window.location.search == '?editonly=1' || window.innerWidth < 1024; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * @returns | ||||
|  * Show the editor. | ||||
|  */ | ||||
| async function edit() { | ||||
| 	if (editing()) { | ||||
| @@ -904,7 +905,7 @@ async function edit() { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * Open a performance trace. | ||||
|  */ | ||||
| function trace() { | ||||
| 	window.open(`/speedscope/#profileURL=${encodeURIComponent('/trace')}`); | ||||
| @@ -982,7 +983,7 @@ async function load(path) { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * Hide the editor. | ||||
|  */ | ||||
| function closeEditor() { | ||||
| 	window.localStorage.setItem('editing', '0'); | ||||
| @@ -990,14 +991,6 @@ function closeEditor() { | ||||
| 	document.getElementById('viewPane').style.display = 'flex'; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * @returns | ||||
|  */ | ||||
| function explodePath() { | ||||
| 	return /^\/~([^\/]+)\/([^\/]+)(.*)/.exec(window.location.pathname); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Save the app. | ||||
|  * @param save_to An optional path to which to save the app. | ||||
| @@ -1111,7 +1104,7 @@ function save(save_to) { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * Prompt to set the app icon. | ||||
|  */ | ||||
| function changeIcon() { | ||||
| 	let value = prompt('Enter a new app icon emoji:'); | ||||
| @@ -1122,7 +1115,7 @@ function changeIcon() { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * Prompt to delete the current app. | ||||
|  */ | ||||
| function deleteApp() { | ||||
| 	let name = document.getElementById('name'); | ||||
| @@ -1143,8 +1136,8 @@ function deleteApp() { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * @returns | ||||
|  * Get the current app URL. | ||||
|  * @return The app URL. | ||||
|  */ | ||||
| function url() { | ||||
| 	let hash = window.location.href.indexOf('#'); | ||||
| @@ -1162,8 +1155,8 @@ function url() { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * @returns | ||||
|  * Get the window hash without the lone '#' if it is empty. | ||||
|  * @return The hash. | ||||
|  */ | ||||
| function hash() { | ||||
| 	return window.location.hash != '#' ? window.location.hash : ''; | ||||
| @@ -1188,7 +1181,7 @@ function api_postMessage(message) { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * Show an error. | ||||
|  * @param error The error. | ||||
|  */ | ||||
| function api_error(error) { | ||||
| @@ -1293,7 +1286,7 @@ function api_requestPermission(permission, id) { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * Log from the app to the console. | ||||
|  */ | ||||
| function api_print() { | ||||
| 	console.log('app>', ...arguments); | ||||
| @@ -1308,7 +1301,7 @@ function api_setHash(hash) { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * Process an incoming WebSocket message. | ||||
|  * @param message The message. | ||||
|  */ | ||||
| function _receive_websocket_message(message) { | ||||
| @@ -1432,14 +1425,14 @@ function send(value) { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * Notify the app of the window hash changing. | ||||
|  */ | ||||
| function hashChange() { | ||||
| 	send({event: 'hashChange', hash: window.location.hash}); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * Make sure the app is connected on window focus, and notify the app. | ||||
|  */ | ||||
| function focus() { | ||||
| 	if (gSocket && gSocket.readyState == gSocket.CLOSED) { | ||||
| @@ -1450,7 +1443,7 @@ function focus() { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * Notify the app of lost focus. | ||||
|  */ | ||||
| function blur() { | ||||
| 	if (gSocket && gSocket.readyState == gSocket.OPEN) { | ||||
| @@ -1617,7 +1610,7 @@ function openFile(name) { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * Refresh the files list. | ||||
|  */ | ||||
| function updateFiles() { | ||||
| 	let files = document.getElementsByTagName('tf-files-pane')[0]; | ||||
| @@ -1650,7 +1643,7 @@ function makeNewFile(name) { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * Prompt to create a new file. | ||||
|  */ | ||||
| function newFile() { | ||||
| 	let name = prompt('Name of new file:', 'file.js'); | ||||
| @@ -1660,7 +1653,7 @@ function newFile() { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * Prompt to remove a file. | ||||
|  */ | ||||
| function removeFile() { | ||||
| 	if (confirm('Remove ' + gCurrentFile + '?')) { | ||||
| @@ -1670,7 +1663,7 @@ function removeFile() { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * Export the app to a zip file, which is downloaded by the browser. | ||||
|  */ | ||||
| async function appExport() { | ||||
| 	let JsZip = (await import('/static/jszip.min.js')).default; | ||||
| @@ -1728,7 +1721,7 @@ async function save_file_to_blob_id(name, file) { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * Prompt to import an app from a zip file. | ||||
|  */ | ||||
| async function appImport() { | ||||
| 	let JsZip = (await import('/static/jszip.min.js')).default; | ||||
| @@ -1855,7 +1848,9 @@ function toggleVisibleWhitespace() { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // TODOC | ||||
| /** | ||||
|  * Register event handlers and connect the WebSocket on load. | ||||
|  */ | ||||
| window.addEventListener('load', function () { | ||||
| 	window.addEventListener('hashchange', hashChange); | ||||
| 	window.addEventListener('focus', focus); | ||||
|   | ||||
							
								
								
									
										34
									
								
								core/http.js
									
									
									
									
									
								
							
							
						
						
									
										34
									
								
								core/http.js
									
									
									
									
									
								
							| @@ -1,8 +1,14 @@ | ||||
| /** | ||||
|  * TODOC | ||||
|  * TODO: document so we can improve this | ||||
|  * @param {*} url | ||||
|  * @returns | ||||
|  * \file | ||||
|  * \defgroup tfhttp Tilde Friends HTTP Client JS | ||||
|  * Tilde Friends server-side HTTP client. | ||||
|  * @{ | ||||
|  */ | ||||
|  | ||||
| /** | ||||
|  * Parse a URL into protocol, host, path, and port parts. | ||||
|  * @param url | ||||
|  * @return An object of the URL parts. | ||||
|  */ | ||||
| function parseUrl(url) { | ||||
| 	// XXX: Hack. | ||||
| @@ -16,9 +22,9 @@ function parseUrl(url) { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * @param {*} data | ||||
|  * @returns | ||||
|  * Parse an HTTP response into headers and body content. | ||||
|  * @param data The response data, headers and body included. | ||||
|  * @return headers and body data. | ||||
|  */ | ||||
| function parseResponse(data) { | ||||
| 	let firstLine; | ||||
| @@ -36,15 +42,15 @@ function parseResponse(data) { | ||||
| 			headers[line.substring(colon)] = line.substring(colon + 1); | ||||
| 		} | ||||
| 	} | ||||
| 	return {body: data}; | ||||
| 	return {headers: headers, body: data}; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * @param {*} url | ||||
|  * @param {*} options | ||||
|  * @param {*} allowed_hosts | ||||
|  * @returns | ||||
|  * Make an HTTP request. | ||||
|  * @param url The URL. | ||||
|  * @param options Request options. | ||||
|  * @param allowed_hosts List of allowed hosts. | ||||
|  * @return A promise resolved with the response headers and body. | ||||
|  */ | ||||
| export function fetch(url, options, allowed_hosts) { | ||||
| 	let parsed = parseUrl(url); | ||||
| @@ -111,3 +117,5 @@ export function fetch(url, options, allowed_hosts) { | ||||
| 			}); | ||||
| 	}); | ||||
| } | ||||
|  | ||||
| /** @} */ | ||||
|   | ||||
| @@ -15,8 +15,8 @@ let g_next_id = 1; | ||||
| let g_calls = {}; | ||||
|  | ||||
| /** | ||||
|  * TODOC | ||||
|  * @returns | ||||
|  * Check if being called from a browser vs. server-side. | ||||
|  * @return true if called from a browser. | ||||
|  */ | ||||
| function get_is_browser() { | ||||
| 	try { | ||||
|   | ||||
| @@ -25,14 +25,14 @@ | ||||
| }: | ||||
| pkgs.stdenv.mkDerivation rec { | ||||
|   pname = "tildefriends"; | ||||
|   version = "0.0.32"; | ||||
|   version = "0.0.33"; | ||||
|  | ||||
|   src = pkgs.fetchFromGitea { | ||||
|     domain = "dev.tildefriends.net"; | ||||
|     owner = "cory"; | ||||
|     repo = "tildefriends"; | ||||
|     rev = "v${version}"; | ||||
|     hash = "sha256-Dk0NOEQIg2LeENySK0+MgpZEtfsClGq6dZL+eOOpE0U="; | ||||
|     hash = "sha256-9D28gmaBTRVyXhY3zZd/W9PsXA1YZt/K69hz41aVP04="; | ||||
|     fetchSubmodules = true; | ||||
|   }; | ||||
|  | ||||
|   | ||||
							
								
								
									
										2
									
								
								deps/openssl_src
									
									
									
									
										vendored
									
									
								
							
							
								
								
								
								
								
							
						
						
									
										2
									
								
								deps/openssl_src
									
									
									
									
										vendored
									
									
								
							 Submodule deps/openssl_src updated: aea7aaf2ab...0893a62353
									
								
							
							
								
								
									
										2
									
								
								deps/speedscope/index.html
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								deps/speedscope/index.html
									
									
									
									
										vendored
									
									
								
							| @@ -11,7 +11,7 @@ | ||||
|     <link rel="icon" type="image/x-icon" href="favicon-FOKUP5Y5.ico"> | ||||
|   </head> | ||||
|   <body> | ||||
|     <script src="speedscope-7YPLLUY2.js"></script> | ||||
|     <script src="speedscope-HCR63FMT.js"></script> | ||||
|      | ||||
|      | ||||
|      | ||||
|   | ||||
							
								
								
									
										6
									
								
								deps/speedscope/release.txt
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								deps/speedscope/release.txt
									
									
									
									
										vendored
									
									
								
							| @@ -1,3 +1,3 @@ | ||||
| speedscope@1.23.0 | ||||
| Sun Jul  6 20:04:28 PDT 2025 | ||||
| aa9bef50789a2989746b576fff182b6f01dfce6a | ||||
| speedscope@1.23.1 | ||||
| Mon Aug 11 11:43:09 PDT 2025 | ||||
| 0cec0f82c334aed6cf19d43cabeadcda0d95e0fc | ||||
|   | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										6
									
								
								flake.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										6
									
								
								flake.lock
									
									
									
										generated
									
									
									
								
							| @@ -20,11 +20,11 @@ | ||||
|     }, | ||||
|     "nixpkgs": { | ||||
|       "locked": { | ||||
|         "lastModified": 1750622754, | ||||
|         "narHash": "sha256-kMhs+YzV4vPGfuTpD3mwzibWUE6jotw5Al2wczI0Pv8=", | ||||
|         "lastModified": 1753749649, | ||||
|         "narHash": "sha256-+jkEZxs7bfOKfBIk430K+tK9IvXlwzqQQnppC2ZKFj4=", | ||||
|         "owner": "NixOS", | ||||
|         "repo": "nixpkgs", | ||||
|         "rev": "c7ab75210cb8cb16ddd8f290755d9558edde7ee1", | ||||
|         "rev": "1f08a4df998e21f4e8be8fb6fbf61d11a1a5076a", | ||||
|         "type": "github" | ||||
|       }, | ||||
|       "original": { | ||||
|   | ||||
| @@ -1,8 +1,8 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <manifest xmlns:android="http://schemas.android.com/apk/res/android" | ||||
| 	package="com.unprompted.tildefriends" | ||||
| 	android:versionCode="40" | ||||
| 	android:versionName="0.0.33"> | ||||
| 	android:versionCode="41" | ||||
| 	android:versionName="0.2025.8-wip"> | ||||
| 	<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/> | ||||
| 	<uses-permission android:name="android.permission.INTERNET"/> | ||||
| 	<application | ||||
|   | ||||
| @@ -81,14 +81,14 @@ public class TildeFriendsActivity extends Activity { | ||||
|  | ||||
| 		TildeFriendsActivity activity = this; | ||||
|  | ||||
| 		Log.w("tildefriends", "Watching for changes in: " + getFilesDir().toString()); | ||||
| 		observer = make_file_observer(getFilesDir().toString(), port_file_path); | ||||
| 		observer.startWatching(); | ||||
|  | ||||
| 		set_status("Starting server..."); | ||||
| 		server_thread = new Thread(new Runnable() { | ||||
| 			@Override | ||||
| 			public void run() { | ||||
| 				Log.w("tildefriends", "Watching for changes in: " + getFilesDir().toString()); | ||||
| 				observer = make_file_observer(getFilesDir().toString(), port_file_path); | ||||
| 				observer.startWatching(); | ||||
|  | ||||
| 				Log.w("tildefriends", "Calling tf_server_main."); | ||||
| 				int result = tf_server_main( | ||||
| 					getFilesDir().toString(), | ||||
| @@ -426,7 +426,7 @@ public class TildeFriendsActivity extends Activity { | ||||
| 				Log.w("tildefriends", "onServiceDisconnected"); | ||||
| 			} | ||||
| 		}; | ||||
| 		s_activity.bindService(intent, s_activity.service_connection, BIND_AUTO_CREATE); | ||||
| 		s_activity.bindService(intent, s_activity.service_connection, BIND_AUTO_CREATE | BIND_NOT_FOREGROUND); | ||||
| 	} | ||||
|  | ||||
| 	public static void stop_sandbox() { | ||||
| @@ -445,6 +445,7 @@ public class TildeFriendsActivity extends Activity { | ||||
| 				hide_status(); | ||||
| 				web_view.loadUrl(base_url + "login/auto"); | ||||
| 			}); | ||||
| 			observer.stopWatching(); | ||||
| 			observer = null; | ||||
| 		} else { | ||||
| 			runOnUiThread(() -> { | ||||
|   | ||||
| @@ -13,13 +13,13 @@ | ||||
| 	<key>CFBundlePackageType</key> | ||||
| 	<string>APPL</string> | ||||
| 	<key>CFBundleShortVersionString</key> | ||||
| 	<string>0.0.33</string> | ||||
| 	<string>0.2025.8</string> | ||||
| 	<key>CFBundleSupportedPlatforms</key> | ||||
| 	<array> | ||||
| 		<string>iPhoneOS</string> | ||||
| 	</array> | ||||
| 	<key>CFBundleVersion</key> | ||||
| 	<string>15</string> | ||||
| 	<string>16</string> | ||||
| 	<key>DTPlatformName</key> | ||||
| 	<string>iphoneos</string> | ||||
| 	<key>LSRequiresIPhoneOS</key> | ||||
|   | ||||
| @@ -1,2 +1,2 @@ | ||||
| #define VERSION_NUMBER "0.0.33" | ||||
| #define VERSION_NUMBER "0.2025.8-wip" | ||||
| #define VERSION_NAME "This program kills fascists." | ||||
|   | ||||
		Reference in New Issue
	
	Block a user