From 56a9a1ccf764ff7a6af8f8a2292cb07443cb4aa7 Mon Sep 17 00:00:00 2001
From: Jake Vanderwerf <get@jakevanderwerf.ca>
Date: Thu, 28 May 2026 18:19:57 +0000
Subject: [PATCH] =New Gitbit setpu

---
 assets/js/concise/CRUD.js | 2310 ++++++++++++++++++++++++++++++++++++++++++++--------------
 1 files changed, 1,741 insertions(+), 569 deletions(-)

diff --git a/assets/js/concise/CRUD.js b/assets/js/concise/CRUD.js
index d0154d3..82b8b4c 100644
--- a/assets/js/concise/CRUD.js
+++ b/assets/js/concise/CRUD.js
@@ -1,551 +1,1447 @@
-/**
- * Main CRUD Manager - Coordinates everything
- */
 class CRUDManager {
-	constructor(config) {
+	constructor(){
+		this.container = document.querySelector('.crud[data-content]:not([data-ignore])');
+		if (!this.container) return;
+		this.content = this.container.dataset.content;
+		this.endpoint = this.container.dataset.endpoint??'content';
+		this.singular = this.container.dataset.singular;
+		this.plural = this.container.dataset.plural;
+
 		this.queue = window.jvbQueue;
-		this.config = config;
-		this.content = config.content || false;
-		this.settings = window.jvbUserSettings;
 		this.a11y = window.jvbA11y;
-		if (!this.content) {
-			return;
-		}
+		this.error = window.jvbError;
+		this.populate = window.jvbPopulate;
+		this.cache = new window.jvbCache(this.content);
+
+		this.activeItem = null;
 		this.isTimeline = false;
-		this.currentItemID = null;
+		this.isPopulating = false;
 
-		this.initElements();
-		this.updateBulkOptions();
-
-		// Initialize components
-		const store = window.jvbStore.register(
-			this.content,
-			{
-				storeName: this.content,
-				keyPath: 'id',
-				endpoint: 'content',
-				headers: {
-					'action_nonce': window.auth.getNonce('dash'),
-				},
-				indexes: [
-					{name: 'id', keyPath: 'id'},
-					{ name: 'status', keyPath: 'status'},
-					{ name: 'date', keyPath: 'date'},
-					{ name: 'modified', keyPath: 'modified'},
-					{ name: 'title', keyPath: 'title'}
-				],
-				filters: {
-					content: this.content,
-					user: window.auth.getUser(),
-					page: 1,
-					status: 'all',
-					orderby: 'modified', //or title
-					order: 'desc'
-				},
-				TTL: 30 * 60 * 1000,	//30 minutes cache
-				showLoading: true,
-			});
-		this.store = store[this.content];
-
-		this.status = 'all';
-		this.filterTimeout = null;
-
-		this.viewController = new window.jvbViews(this.ui.container, this.store);
-		this.tableForm = null;
-		this.tableChanges = new Map();
-
-
-		this.formController = (this.isTimeline) ? new window.jvbForm({collectFormData: () => this.collectTimelineData.bind(this)}) : new window.jvbForm();
-		this.viewController.subscribe((event, form) => {
-			if (event === 'table-view' && !this.tableForm) {
-				if (!this.tableForm) {
-					this.tableForm = this.formController.registerForm(form, {
-						autosave: false,
-						formStatus: false,
-						isTable: true,
-					});
-				}
-
-			} else if (event === 'not-table-view') {
-				if (this.tableForm) {
-
-				}
-			} else if (event === 'order-changed') {
-				let data = this.store.get(form);
-				if (!data) {
-					return;
-				}
-				let changes = {};
-				changes[form] = data;
-				this.savePosts(changes, `Updating progression order`);
-			}
-		});
-
-		this.formController.subscribe((event, data) => {
-			switch(event) {
-				case 'form-submit':
-				case 'form-autosave':
-					this.handleFormChange(event,data);
-					break;
-			}
-		});
-
-		this.queue.subscribe((event, data) => {
-			if (!Object.hasOwn(data, 'endpoint') || !['content', 'uploads/groups'].includes(data.endpoint)) return;
-			if (event === 'operation-completed') {
-				this.handleQueueSuccess(event, data);
-			} else if (event === 'operation-failed-permanent') {
-				this.handleQueueFailure(event, data);
-			}
-		});
-
-
-		// Track initialization
-		this.initialized = false;
+		//Track Changes
+		this.changes = new Map();
+		this.items = new Map();
 
 		this.init();
 	}
-	handleFormChange(event, data) {
-		let title = data.fullData.post_title;
-		let changes = (Object.hasOwn(data, 'changes')) ? data.changes : data.fullData;
 
-		let theChanges = {};
-		if (this.isTimeline) {
-			theChanges[this.currentItemID] = changes;
-			this.savePosts(theChanges, title);
-			return;
+	init() {
+		this.initElements();
+		this.initListeners();
+		this.defineTemplates();
+		let cached = this.initSettings();
+		this.initStore(cached);
+		this.checkHideFilters();
+		this.initIntegrations();
+		this.initUploader();
+		this.initModals();
+	}
+
+	defineTemplates() {
+		const T = window.jvbTemplates;
+		const crud = this;
+
+		const baseSetup = (el, refs, data) => {
+			el.dataset.itemId = data.id;
+			let wrapper = refs.checkbox.closest('.preview');
+			window.prefixInput(refs.checkbox, `select-${data.id}`, wrapper, true);
+			refs.checkbox.value = data.id;
+			refs.checkbox.checked = crud.selected.has(parseInt(data.id));
+			if (refs.selectLabel) refs.selectLabel.htmlFor = `select-${data.id}`;
+
+			if (refs.edit) refs.edit.dataset.id = data.id;
+			if (refs.trash) refs.trash.dataset.id = data.id;
+		};
+		const imageSetup = function(el, refs, data) {
+			let hasThumbnail = data?.fields?.post_thumbnail || data?.fields?.thumbnail;
+			if (hasThumbnail) {
+				const thumbnail = data.images[hasThumbnail] ?? {};
+				refs.img.src = thumbnail.medium??'';
+				refs.img.alt = thumbnail.alt??data.fields.post_title??'';
+			}
+
 		}
 
-		let itemsToRemove = [];
-		switch (true) {
-			case data.config.element === this.ui.forms.edit:
-				theChanges[this.currentItemID] = changes;
-				title = `Saving ${title} Changes`;
-				// Check if status change requires removal
-				if (changes.post_status && this.shouldRemoveItem(changes.post_status)) {
-					itemsToRemove.push(this.currentItemID);
-				}
-				break;
-			case data.config.element === this.ui.forms.bulkEdit:
-				let selected = data.config.element.querySelectorAll('.selected input:checked');
-				selected.forEach(sel => {
-					theChanges[sel.value] = changes;
-					// Check if status change requires removal
-					if (changes.post_status && this.shouldRemoveItem(changes.post_status)) {
-						itemsToRemove.push(sel.value);
+		T.define('gridView', {
+			refs: {
+				img: 'img',
+				checkbox: '.select-item',
+				selectLabel: 'label.select-item-label',
+				edit: '[data-action="edit"]',
+				trash: '[data-action="trash"]'
+			},
+			setup({ el, refs, manyRefs, data }) {
+				baseSetup(el, refs, data);
+				imageSetup(el, refs, data);
+			}
+		});
+
+		T.define('listView', {
+			refs: {
+				img: 'img',
+				checkbox: '.select-item',
+				selectLabel: 'label.select-item-label',
+				edit: '[data-action="edit"]',
+				trash: '[data-action="trash"]'
+			},
+			manyRefs: {
+				attrs: '[data-attr]',
+				fields: '[data-field]'
+			},
+			setup({ el, refs, manyRefs, data }) {
+				baseSetup(el, refs, data);
+				imageSetup(el, refs, data);
+				manyRefs?.attrs?.forEach(el => {
+					const value = data[el.dataset.attr];
+					if (value && value !=='') {
+						el.textContent = value;
+					} else {
+						el.remove();
+					}
+				});
+				manyRefs?.fields?.forEach(el => {
+					const value = data.fields?.[el.dataset.field];
+					if (value && value !== '') {
+						el.tagName === 'DIV' ? el.innerHTML = value : el.textContent = value;
+					} else {
+						el.remove();
+					}
+				});
+			}
+		});
+
+		let tableRefs = {};
+		let tableMany = {};
+		if (this.isTimeline) {
+			tableRefs.sharedRow = 'tr.shared';
+			tableRefs.point = 'tr.timeline-point';
+		}
+		T.define('tableView', {
+			refs: {
+				checkbox: '.select-item',
+				selectLabel: 'label.select-item-label',
+				... tableRefs,
+			},
+			manyRefs: {
+				inputs: 'input,select,textarea',
+				status: 'input[name="post_status"]',
+				selectors: '[data-type="selector"]',
+				fields: '[data-field]',
+				... tableMany,
+			},
+			setup({ el, refs, manyRefs, data }) {
+				baseSetup(el, refs, data);
+
+				manyRefs?.inputs?.forEach(el => {
+					let wrapper = el.closest('[data-field]');
+					window.prefixInput(el, `${data.id}-`, wrapper);
+				});
+
+				manyRefs?.status?.forEach(el => {
+					if (el.value === data.status) {
+						el.checked = true;
 					}
 				});
 
-				title = `Updating ${selected.length} ${this.config.plural??'posts'} Changes`;
-				break;
-			case data.config.element === this.ui.forms.create:
-				if (event === 'form-submit') {
-					theChanges[data.config.data['form-id']] = changes;
-					title = `Saving ${title} Changes`;
-				}
-				break;
-		}
+				if (crud.isTimeline) {
+					if (refs.sharedRow) {
+						refs.sharedRow.querySelectorAll('input,select,textarea').forEach(input => {
+							let wrapper = input.closest('[data-field]');
+							window.prefixInput(input, `${data.id}-`, wrapper);
+						});
 
-		// Handle visual removal with stagger effect
-		if (itemsToRemove.length > 0) {
-			let delay = 0;
-			itemsToRemove.forEach(itemId => {
-				setTimeout(() => {
-					const element = document.querySelector(`.item[data-id="${itemId}"]`);
-					if (element) {
-						window.fade(element, false);
+						crud.populate.populate(refs.sharedRow, data);
+
+						// Handle status radios in shared row
+						refs.sharedRow.querySelectorAll('input[name="post_status"]').forEach(el => {
+							if (el.value === data.status) {
+								el.checked = true;
+							}
+						});
 					}
-				}, delay);
-				delay += 50; // Stagger by 50ms
-			});
 
-			// Clear selection after bulk edit with staggered removal
-			if (data.config.element === this.ui.forms.bulkEdit) {
-				setTimeout(() => {
-					this.viewController.clearSelection();
-				}, delay + 100);
-			}
-		}
+					if (refs.point && data.fields?.timeline) {
 
-		if (Object.keys(theChanges).length === 0) {
-			return;
-		}
+						Object.entries(data.fields.timeline).forEach(([nuthing, timeline], index) => {
+							const point = refs.point.cloneNode(true);
+							point.dataset.index = `${index}`;
+							point.dataset.itemId = timeline.id;
 
-		this.savePosts(theChanges, title);
-	}
+							point.querySelectorAll('input,select,textarea').forEach(input => {
+								let wrapper = input.closest('[data-field]');
+								window.prefixInput(input, `${timeline.id}-`, wrapper);
+							});
 
-	shouldRemoveItem(newStatus) {
-		return (this.status === 'all' && !['publish', 'draft'].includes(newStatus)) ||
-			(newStatus !== this.status);
-	}
+							crud.populate.populate(point, {
+								fields: timeline,
+								images: data.images,
+								taxonomies: data.taxonomies
+							});
 
-	savePosts(changes, title) {
-		if (Object.keys(changes).length === 0) {
-			return;
-		}
+							const imgData = data.images?.[timeline.post_thumbnail];
+							if (imgData) {
+								point.querySelector('.field.upload')?.setAttribute('title', imgData['image-title']??'');
+							}
+							el.insertBefore(point, refs.point);
+						});
+						refs.point.remove();
+					}
+				} else {
+					if (crud.ui.table.form?.dataset.edit !== undefined) {
+						// Non-timeline: prefix all inputs normally
+						manyRefs?.inputs?.forEach(input => {
+							let wrapper = input.closest('[data-field]');
+							window.prefixInput(input, `${data.id}-`, wrapper);
+						});
 
-		//ensure content is in each post
-		for (let postId in changes) {
-			if (!changes[postId]['content']) {
-				changes[postId]['content'] = this.content;
-			}
-		}
-		let operation = {
-			endpoint: 'content',
-			headers: {
-				'action_nonce': window.auth.getNonce('dash'),
-			},
-			data: {
-				posts: changes,
-			},
-			popup: `Saving changes`,
-			title: title
-		};
+						manyRefs?.status?.forEach(el => {
+							if (el.value === data.status) {
+								el.checked = true;
+							}
+						});
 
-		this.queue.addToQueue(operation);
-
-	}
-	async handleQueueSuccess(event, data) {
-		this.store.clearCache();
-		this.store.fetch();
-	}
-	handleQueueFailure(event, data) {
-		console.error('Operation failed permanently:', data);
-		// Optionally show error notification to user
-		this.a11y?.announce(`Operation failed: ${data.error_message || 'Unknown error'}`);
-	}
-
-	initElements() {
-		this.elements = {
-			modals: {
-				create: 'dialog.create',
-				edit: 'dialog.edit',
-				bulkEdit: 'dialog.bulkEdit'
-			},
-			container: '.crud[data-content]',
-			grid: '.item-grid',
-			bulkSelectActions: '.bulk-action-select',
-			forms: {
-				create: 'dialog.create form',
-				edit: 'dialog.edit form',
-				bulkEdit: 'dialog.bulkEdit form'
-			},
-			uploader: 'details.uploader'
-		};
-		this.ui = window.uiFromSelectors(this.elements);
-		if (this.ui.uploader) {
-			window.jvbUploads.scanFields(document.querySelector(this.elements.uploader));
-
-			window.jvbUploads.subscribe((event, data) => {
-				if (event === 'sent-to-queue') {
-					console.log(data);
-					if (data === this.ui.uploader.querySelector('[data-uploader]')?.dataset.uploader) {
-						window.debouncer.schedule('crud-complete', ()=> {
-							this.store.clearHttpHeaders();
+						crud.populate.populate(el, data);
+					} else {
+						const fields = (Object.hasOwn(data, 'fields')) ? data.fields : data;
+						manyRefs?.fields?.forEach(field => {
+							if (Object.hasOwn(fields, field.dataset.field) && fields[field.dataset.field] !== '') {
+								let value = fields[field.dataset.field];
+								let p = fields.children[0];
+								if (p) {
+									p.textContent = field.dataset.field === 'date'
+										? window.formatTimeAgo(value)
+										: value;
+								}
+							}
 						});
 					}
 				}
+
+				manyRefs?.selectors?.forEach(selector => selector.setAttribute('data-lazy', ''));
+			}
+		});
+
+		T.define('emptyState');
+
+		T.define('bulkItem', {
+			refs: {
+				checkbox: 'input',
+				img: 'img',
+				label: 'label'
+			},
+			setup({el, refs, manyRefs, data}) {
+				if (refs.checkbox) {
+					refs.checkbox.id = `bulk_${data.id}`;
+					refs.checkbox.value = data.id;
+					refs.checkbox.checked = true;
+					refs.checkbox.name ='selected[]';
+				}
+				let thumbnail = data?.images[data?.fields?.post_thumnbail]??{};
+				if (refs.img && Object.keys(thumbnail).length >0) {
+					refs.img.src = thumbnail.medium??'';
+					refs.img.alt = thumbnail.alt??'';
+				}
+
+				if (refs.label) {
+					refs.label.title = item.fields.post_title;
+				}
+			}
+		});
+		T.define('trashOptions');
+		T.define('notTrashOptions');
+		T.define('contentTable');
+	}
+
+	initElements() {
+		this.allowedFilters = ['status', 'orderby', 'order', 'search', 'date-filter', 'dateFrom', 'dateTo'];
+		this.selectors = {
+			buttons: {
+				create: '.create-item',
+				clearFilters: '[data-action="clear-filters"]'
+			},
+			views: {
+				grid: 'input[data-view="grid"]',
+				list: 'input[data-view="list"]',
+				table: 'input[data-view="table"]'
+			},
+			modals: {
+				create: {
+					modal: 'dialog.create',
+					form: 'dialog.create form',
+					h2: 'dialog.create h2',
+				},
+				edit: {
+					modal: 'dialog.edit',
+					form: 'dialog.edit form',
+					h2: 'dialog.edit h2',
+				},
+				bulkEdit: {
+					modal: 'dialog.bulkEdit',
+					selected: 'dialog.bulkEdit .selected',
+					h2: 'dialog.bulkEdit h2 span',
+					form: 'dialog.bulkEdit form'
+				},
+				date: {
+					modal: 'dialog.date-range',
+					start: 'dialog.date-range .date-start',
+					end: 'dialog.date-range .date-end',
+					month: 'dialog.date-range .month-select',
+				}
+			},
+			grid: `.${this.content}.item-grid`,
+			table: {
+				nav: '#vertical',
+				form: 'form.table',
+				table: 'form.table table',
+				body: 'form.table body',
+				head: 'form.table thead',
+				foot: 'form.table tfoot',
+				selectedColumns: '.all-filters .multi-select',
+				columns: 'thead th',
+			},
+			bulk: {
+				action: '.bulk-action-select',
+				count: '.bulk-controls .selected-count',
+				control: '.bulk-controls .bulk-actions',
+				select: '.bulk-controls select',
+				selectAll: '.select-all'
+			},
+			filters: {
+				container: 'details.all-filters',
+				search: '.all-filters input[type="search"]',
+				status: {
+					all: '[name="status"]#all',
+					publish: '[name="status"]#publish',
+					draft: '[name="status"]#draft',
+					trash: '[name="status"]#trash',
+				},
+				orderby: {
+					date: '[name="orderby"]#date',
+					alphabetical: '[name="orderby"]#alphabetical',
+				},
+				order: {
+					asc: '[name="order"][value="asc"]',
+					desc: '[name="order"][value="desc"]'
+				},
+				date: '[data-filter="date"]'
+			},
+			uploader: {
+				details: 'details.uploader',
+				form: 'details.uploader form',
+				uploader: 'details.uploader [data-field-type="upload"]'
+			}
+		}
+
+		this.ui = window.uiFromSelectors(this.selectors);
+		const taxFilters = document.querySelectorAll('[data-filter="taxonomies"]');
+		if (taxFilters.length > 0) {
+			this.ui.filters.taxonomies = {};
+			taxFilters.forEach(tax => {
+				const taxonomy = tax.dataset.taxonomy;
+				this.ui.filters.taxonomies[taxonomy] = tax;
+				this.allowedFilters.push(`tax_${taxonomy}`);
 			});
 		}
 		this.isTimeline = !!document.querySelector('[data-timeline]');
 	}
-	init() {
-		if (this.ui.uploader){
-			this.settings.addSetting(this.ui.uploader, 'open');
-			this.ui.uploader.addEventListener('toggle', (e) =>{
-				this.settings.saveSetting('open', this.ui.uploader.open ? 'on' : 'off');
+		initUploader() {
+			if (!this.ui.uploader.form) return;
+			this.uploadForm = this.forms.registerForm(this.ui.uploader.form).id??false;
+
+			// window.jvbUploads.scanFields(this.ui.uploader);
+			window.jvbUploads.subscribe((event, data) => {
+				if (event === 'sent-to-queue') {
+					if (data.field.id === this.ui.uploader.uploader.dataset.uploader) {
+						if (this.uploadForm ) {
+							this.forms.store.delete(this.uploadForm);
+						}
+
+						window.debouncer.schedule('crud-complete', ()=> {
+							this.store.clearCache();
+						});
+					}
+				}
+
+				if (event === 'sent-to-queue' && data.field) {
+					const fieldName = data.field.config.name;
+					const itemId = data.field.config.itemID;
+					if (itemId && fieldName) {
+						if (this.changes.has(itemId)) {
+							delete this.changes.get(itemId)[fieldName];
+						}
+					}
+				}
 			});
 		}
+		initModals() {
+			this.modals = {};
+			for (let [name, modal] of Object.entries(this.ui.modals)) {
+				if (!modal.modal) continue;
+				this.modals[name] = new window.jvbModal(modal.modal);
 
-		// Set up filter controls
-		this.filterHandler = this.handleFilterChange.bind(this);
-		this.changeHandler = this.handleChange.bind(this);
+				this.modals[name].subscribe((event, data) => {
+					switch (event) {
+						case 'modal-close':
+							const formId = this.ui.modals[name].form.dataset.formId;
+							if (formId) {
+								this.forms.clearForm(formId);
+							}
+
+							this.resetForm(this.ui.modals[name].form);
+
+							if (name === 'date') {
+								this.handleCustomDateSelection()
+							}
+							if (['edit','bulkEdit','create'].includes(name)) {
+								//handle escapes (not form submits)
+								if (window.debouncer.timeouts.has(`save-${this.content}`)) {
+									this.scheduleSave(0);
+								}
+							}
+							break;
+						case 'modal-open':
+
+							break;
+					}
+				})
+			}
+
+		}
+
+	initStore(cached) {
+		let filters = {
+			... this.defaults,
+			...cached
+		};
 
 
+		const stores = window.jvbStore.register(
+			this.content,
+		[
+				{
+					storeName: this.content,
+					keyPath: 'id',
+					endpoint: this.endpoint??'content',	//for taxonomy stores
+					headers: {
+						'X-Action-Nonce': window.auth.getNonce('dash'),
+					},
+					indexes: [
+						{name: 'id', keyPath: 'id'},
+						{ name: 'status', keyPath: 'status'},
+						{ name: 'date', keyPath: 'date'},
+						{ name: 'modified', keyPath: 'modified'},
+						{ name: 'title', keyPath: 'title'},
+					],
+					isAuth: true,
+					filters: filters,
+					ignore: ['content', 'user'],
+					TTL: 60 * 60 * 1000, 		//1 hour cache
+					showLoading: true,
+				},
+				{
+					storeName: 'changes',
+					keyPath: 'id'
+				}
+			]
+		);
 
-		this.modals = {};
-		for (let [name, modal] of Object.entries(this.ui.modals)) {
-			this.modals[name] = new window.jvbModal(modal);
+		this.changesStore = stores['changes'];
+		this.store = stores[this.content];
 
-			this.modals[name].subscribe((event, data) => {
-				switch (event) {
-					case 'modal-close':
-						this.currentItemID = null;
-						this.formController.cleanupForm(this.modals[name].modal.querySelector('form').dataset.formId);
-						//double check we have finished saving
-						break;
-					case 'modal-open':
-						//probably not needed in this class
-						break;
+		this.store.subscribe((event, data) => {
+			switch (event) {
+				case 'data-loaded':
+					this.render();
+					this.selectionHandler.collectItems();
+					break;
+			}
+		});
+
+		this.changesStore.subscribe((event, data) => {
+			switch (event) {
+				case 'data-ready':
+					let changes = this.changesStore.getAll();
+					if (changes.length > 0) {
+						changes.forEach(change => {
+							this.changes.set(change.id, change);
+						});
+						this.savePosts('', false).then(()=>{});
+					}
+					break;
+			}
+		});
+	}
+	initIntegrations() {
+		this.selected = new Set();
+		this.selectionHandler = new window.jvbHandleSelection(this.container, {
+			selectAll: {
+				checkbox: '#select-all',
+				label: '.bulk-select label',
+				span: '.bulk-select label span'
+			},
+			wrapper: {
+				wrapper: '.wrap'
+			},
+			item: {
+				idAttribute: 'itemId'
+			}
+		});
+		this.selectionHandler.subscribe((event, data) => {
+			this.selected = new Set([...data.selectedItems].map(id => parseInt(id)));
+			this.ui.bulk.control.hidden = this.selected.size === 0;
+			this.ui.bulk.count.hidden = this.selected.size === 0;
+			this.ui.bulk.count.textContent = `${this.selected.size} ${this.plural} selected`;
+		});
+
+		this.forms = window.jvbForm;
+
+		// this.forms.subscribe((event, data) => {
+		// 	switch(event) {
+		// 		case 'form-submit':
+		// 		case 'form-autosave':
+		// 			// this.handleFormChange(event,data);
+		// 			break;
+		// 	}
+		// });
+
+		if (window.jvbUploads) {
+			window.jvbUploads.subscribe((event, data) => {
+				if (event === 'groups_uploaded' && data.content === this.content) {
+					this.handleGroupsUploaded(data);
 				}
 			});
 		}
 
-		// Set up global event delegation
-		this.setupEventDelegation();
+		this.queue.subscribe((event, data) => {
+			if (['image_upload', 'video_upload', 'document_upload'].includes(data.type)
+				&& event === 'operation-status'
+				&& data.status === 'completed') {
+				this.store.clearCache();
+			}
 
-		this.setupFilters();
 
-		this.initialized = true;
-	}
+			if (event === 'operation-status'
+				&& data.status === 'completed'
+				&& data.endpoint === 'uploads/groups') {
+				if (data.result && data.result.group_mappings) {
+					console.log('Handling group mapping from queue response');
+					this.handleGroupMappings(data.result.group_mappings);
+				}
 
-	setupEventDelegation() {
-		document.addEventListener('change', this.changeHandler);
-		// Single event listener for all CRUD actions
-		document.addEventListener('click', (e) => {
-			// Check for action buttons
-			const actionBtn = e.target.closest('[data-action]');
-			if (actionBtn) {
-				e.preventDefault();
-				const action = actionBtn.dataset.action;
-				const id = actionBtn.dataset.id;
+				this.store.clearCache();
+			}
 
-				switch(action) {
-					case 'edit':
-						this.populateEditForm(id);
-						this.modals.edit.handleOpen();
-						break;
+			if (event === 'operation-status'
+				&& data.status === 'completed'
+				&& data.type === 'content_update') {
 
-					case 'delete':
-						if (confirm('Delete this item?')) {
-							let changes = {};
-							changes[actionBtn.dataset.id] = {
-								'post_status': 'delete',
-								'content': this.content
-							};
-							window.fade(actionBtn.closest('.item'), false);
-							this.savePosts(changes, `Sending ${this.singular} to trash...`);
-							this.store.delete(id);
-						}
-						break;
-					case 'trash':
-						let changes = {};
-						changes[actionBtn.dataset.id] = {
-							'post_status': 'trash',
-							'content': this.content
-						};
-						window.fade(actionBtn.closest('.item'), false);
-						this.savePosts(changes, `Sending ${this.singular} to trash...`);
-						break;
+				this.store.clearCache();
 
-					case 'create':
-						this.modals.create.dataset.itemId = 'new';
-						this.modals.create.dataset.content = this.content;
-						this.modals.create.handleOpen();
-						break;
+				if (!data.result || !data.result.success || !data.result.errors)
+				{
+					console.warn('Content update completed but no results', data);
+					return;
+				}
 
-					case 'bulk-edit':
-						const selected = Array.from(this.viewController.selectedItems);
-						if (selected.length > 0) {
+				if (Object.keys(data.result.success).length > 0) {
+					this.checkCompletedChanges(Object.entries(data.result.success));
+				}
+				if (Object.keys(data.result.errors).length > 0) {
+					this.checkFailedChanges(Object.entries(data.result.errors));
+					return;
+				}
 
-							this.modals.bulkEdit.handleOpen();
-						}
-						break;
+				if (Object.keys(data.result.success).length === 0) {
+					console.log(data.result.success);
+					data.result.success.forEach(id => this.changesStore.delete(id));
 
-					case 'bulk-delete':
-						const toDelete = Array.from(this.viewController.selectedItems);
-						if (toDelete.length > 0 && confirm(`Delete ${toDelete.length} items?`)) {
-							toDelete.forEach(id => this.store.delete(id));
-							this.viewController.clearSelection();
-						}
-						break;
-
-					case 'sync':
-						// this.store.syncQueue();
-						break;
-
-					case 'refresh':
-						this.store.fetch();
-						break;
+					this.store.clearCache();
 				}
 			}
 
-			let createButton = e.target.closest('.create-item');
-			if (createButton) {
-				this.formController.registerForm(this.ui.forms.create);
-				this.modals.create.handleOpen();
-			}
+			if (event === 'sent-to-server' && data.type === 'content_update') {
+				if (data instanceof FormData) return;
 
-			let clearSelection = e.target.closest('.cancel-bulk');
-			if (clearSelection) {
-				this.viewController.selectAll(false);
-			}
-		});
-
-		// Keyboard shortcuts
-		document.addEventListener('keydown', (e) => {
-			// Ctrl/Cmd + A to select all
-			if ((e.ctrlKey || e.metaKey) && e.key === 'a') {
-				if (this.ui.container && this.ui.container.contains(document.activeElement)) {
-					e.preventDefault();
-					this.viewController.selectAll();
+				for ( let [id, changes] of Object.entries(data.posts)) {
+					this.compareStored(id, changes);
 				}
 			}
 
-			// ESC to clear selection
-			if (e.key === 'Escape' && this.viewController?.selectedItems.size > 0 && window.jvbModal.getAllModals().length === 0) {
-				this.viewController.clearSelection();
-			}
 		});
 	}
-	handleChange(e) {
-		if (e.target.closest('[data-id]')) {
-			if (this.isTimeline) {
-				this.handleTimelineTableChange(e);
+	checkCompletedChanges(items) {
+		for (let [id, data] of items) {
+			this.compareStored(id, data);
+		}
+	}
+		compareStored(id, data) {
+			let stored = this.changesStore.get(id);
+			if (!stored) return;
+
+			for (let [field, value] of Object.entries(data)) {
+				if (Object.hasOwn(stored, field)) {
+					let changes = window.getDifferences.map(stored[field], value);
+					if (!changes) {
+						delete stored[field];
+					} else {
+						stored[field] = changes;
+					}
+				}
+			}
+
+			let hasID = Object.hasOwn(stored, 'id');
+			let hasContent = Object.hasOwn(stored, 'content');
+			if ((hasID && hasContent && Object.keys(stored).length === 2)
+				|| ((hasID || hasContent) && Object.keys(stored).length === 1)
+				|| Object.keys(stored).length === 0
+			) {
+				this.changesStore.delete(id);
+				this.store.clearCache();
 			} else {
-				this.handleTableChange(e);
+				this.changesStore.save(stored);
 			}
+		}
+	checkFailedChanges(items) {
+		//TODO do something.
+	}
+
+	initSettings() {
+		this.defaults = {
+			content: this.content,
+			user: window.auth.getUser(),
+			page: 1,
+			status: 'all',
+			orderby: 'date',
+			order: 'desc',
+			search: '',
+		}
+
+		let updateFilters = {};
+		//current view (defaults to grid)
+		let defaultView = this.container.dataset.view??'grid'
+		this.view = this.cache.get('view')??defaultView;
+		if (this.view !== defaultView) {
+			this.ui.views[this.view].checked = true;
+		}
+		//current status (defaults to all)
+		this.status = this.cache.get('status')??this.defaults.status;
+		if (this.status !== this.defaults.status) {
+			this.ui.filters.status[this.status].checked = true;
+			updateFilters.status = this.status;
+		}
+		//orderby & order
+		this.orderby = this.cache.get('orderby')??this.defaults.orderby;
+		if (this.orderby !== this.defaults.orderby) {
+			this.ui.filters.orderby[this.orderby].checked = true;
+			updateFilters.orderBy = this.orderby;
+		}
+		this.order = this.cache.get('order')??this.defaults.order;
+		if (this.order !== this.defaults.order) {
+			this.ui.filters.order[this.order].checked = true;
+			updateFilters.order = this.order;
+		}
+
+		if (this.ui.filters.taxonomies) {
+			Object.entries(this.ui.filters.taxonomies).forEach(([taxonomy, element]) => {
+				const filterKey = `tax_${taxonomy}`;
+				const cached = this.cache.get(filterKey);
+				if (cached) {
+					element.value = cached;
+					updateFilters[filterKey] = cached;
+				}
+			});
+		}
+
+		let tabDirection = this.cache.get('tabNav')??'horizontal';
+		if (this.ui.table.nav && tabDirection === 'vertical') {
+			this.ui.table.nav.checked = true;
+		}
+
+
+
+		//Setup details open functionality
+		let details = {
+			showFilters: {
+				element: this.ui.filters.container,
+				default: 'closed',
+			},
+			showUploader: {
+				element: this.ui.uploader.details,
+				default: 'open'
+			}
+		};
+		for (let [name, conf] of Object.entries(details)) {
+			if (conf.element) {
+				let cached = this.cache.get(name)??conf.default;
+				conf.element.open = cached === 'open';
+				conf.element.addEventListener('toggle', ()=> {
+					this.cache.set(name, conf.element.open ? 'open' : 'closed');
+				});
+			}
+		}
+
+		return updateFilters;
+	}
+	/****************************************************************
+	 EVENT LISTENERS
+	****************************************************************/
+	initListeners() {
+		this.changeHandler = this.handleChange.bind(this);
+		this.clickHandler = this.handleClick.bind(this);
+		this.inputHandler = this.handleInput.bind(this);
+		this.submitHandler = this.handleModalSubmit.bind(this);
+
+		document.addEventListener('change', this.changeHandler);
+		document.addEventListener('click', this.clickHandler);
+		if (this.ui.filters.search) {
+			this.ui.filters.search.addEventListener('input', this.inputHandler);
+		}
+
+		for (let [name, modal] of Object.entries(this.ui.modals)) {
+			if (modal.form) {
+				modal.form.addEventListener('submit', this.submitHandler);
+			}
+		}
+	}
+
+	handleModalSubmit(e) {
+		e.preventDefault();
+		const form = e.target;
+		const modal = form.closest('dialog');
+		if (!modal) return;
+
+		if (modal.classList.contains('create')) {
+			this.handleCreateSubmit(modal);
 			return;
 		}
-		if (e.target.classList.contains('bulk-action-select')) {
-			if (e.target.value.startsWith('tax-')) {
-				const taxonomy = e.target.value.replace('tax-', '');
-				this.openTaxonomyModal(taxonomy);
-				e.target.value = '';
+
+		let title = `Saving changes for multiple ${this.plural}`;
+
+		this.scheduleSave(0);
+		this.modals.edit.handleClose();
+	}
+
+	async handleCreateSubmit(modal) {
+		const itemId = modal.dataset.itemId;
+
+		// 1. Flush changes to store
+		if (this.changes.size > 0) {
+			this.cancelBackup();
+			await this.handleBackup();
+		}
+
+		const changes = await this.changesStore.getAll();
+		if (changes.length === 0) return;
+
+		let allChanges = {};
+		changes.forEach(change => {
+			const { id, ...rest } = change;
+			allChanges[id] = rest;
+		});
+
+		// 2. Queue content creation, get operationId
+		let contentOpId = this.queue.addToQueue({
+			endpoint: this.endpoint,
+			headers: {
+				'X-Action-Nonce': window.auth.getNonce('dash'),
+			},
+			data: {
+				posts: allChanges,
+			},
+			popup: `Creating your new ${this.singular}`,
+			title: `Creating your new ${this.singular}`,
+		});
+
+		if (!contentOpId) return;
+
+		// 3. Queue any pending uploads with dependency on content creation
+		const uploadFields = modal.querySelectorAll('[data-upload-field]');
+		for (const fieldEl of uploadFields) {
+			const fieldId = fieldEl.dataset.uploader;
+			if (!fieldId) continue;
+
+			const uploads = window.jvbUploads.stores.uploads.filterByIndex({ field: fieldId });
+			if (uploads.length === 0) continue;
+
+			await window.jvbUploads.queueUploads('uploads', fieldId, contentOpId);
+		}
+	}
+	handleChange(e) {
+		// Early bailout - target must be in an item or be a filter
+		const inItem = e.target.closest('[data-item-id]');
+		const isFilter = e.target.matches('[data-filter]');
+		const isBulkAction = e.target.matches('.bulk-action-select');
+		const isView = e.target.matches('[data-view]');
+
+		if (!inItem && !isFilter && !isBulkAction && !isView) return;
+
+		if (!this.isPopulating && inItem && !e.target.closest('[data-ignore], .select-item')) {
+			this.handleItemUpdate(e);
+			return;
+		}
+
+		if (isView) {
+			this.items.clear();
+			this.handleViewChange(e.target);
+			return;
+		}
+
+		if (isBulkAction) {
+			this.handleBulkAction(e.target);
+			return;
+		}
+
+		if (isFilter) {
+			this.handleFilterChange(e.target);
+			return;
+		}
+
+		// Table-specific handlers
+		if (this.view === 'table') {
+			if (e.target.matches('details.multi-select')) {
+				this.toggleColumn(e.target.id, e.target.checked);
 				return;
 			}
 
-			switch (e.target.value) {
+			if (e.target.matches(this.selectors.table.nav)) {
+				this.tabNav = e.target.checked;
+				this.cache.set('tabNav', e.target.checked ? 'vertical' : 'horizontal');
+			}
+		}
+	}
+		handleBulkAction(bulkAction) {
+			if (bulkAction.value.startsWith('tax-')) {
+				const selectedOption = bulkAction.options[bulkAction.selectedIndex];
+				const taxonomy = selectedOption.dataset.taxonomy;
+				const single = selectedOption.dataset.single;
+				const plural = selectedOption.dataset.plural;
+
+				window.jvbSelector.openEmpty(
+					taxonomy,
+					single,
+					plural,
+					(result) => this.handleBulkTaxonomy(result)
+				);
+				bulkAction.value = '';
+
+				return;
+			}
+			switch(bulkAction.value) {
 				case 'edit':
-					this.populateBulkEdit();
-					this.modals.bulkEdit.handleOpen();
+					this.openBulkEditModal();
 					break;
 				case 'publish':
-					this.setBulkStatus('publish');
+				case 'trash':
+				case 'delete':
+					this.setBulkStatus(bulkAction.value);
 					break;
 				case 'draft':
-					this.setBulkStatus('draft');
-					break;
-				case 'trash':
-					this.setBulkStatus('trash');
-					break;
 				case 'restore':
 					this.setBulkStatus('draft');
 					break;
+			}
+		}
+	handleBulkTaxonomy(result) {
+		if (!result.termIds.length || !this.selected.size) return;
+
+		this.selected.forEach(itemID => {
+			const item = this.store.get(itemID);
+			if (!item) return;
+
+			// Merge existing terms with new ones
+			const existingTerms = item.taxonomies?.[result.taxonomy] || [];
+			const existingIds = existingTerms.map(t => t.id);
+			const newIds = [...new Set([...existingIds, ...result.termIds])];
+
+			this.updateItem(itemID, result.taxonomy, newIds);
+		});
+
+		this.savePosts(`Adding ${result.terms.length} ${result.taxonomy} to ${this.selected.size} ${this.plural}...`,).then(()=> {});
+
+		this.selectionHandler.clearSelection();
+	}
+
+	handleItemUpdate(e) {
+		let item = window.targetCheck(e, '[data-item-id]');
+		if (!item) return;
+
+		// Check if inside a collection field first
+		const collection = e.target.closest('[data-field-type="repeater"], [data-field-type="tag-list"]');
+
+		let name, value;
+		if (collection) {
+			name = collection.dataset.field;
+			value = this.forms.getFieldValue(collection);
+		} else {
+			let field = e.target.closest('[data-field]');
+			name = field.dataset.field;
+			value = this.forms.getFieldValue(e.target);
+		}
+
+		item.dataset.itemId.split(',').forEach(itemId => {
+			this.updateItem(itemId, name, value);
+		});
+	}
+	updateItem(itemId, name, value) {
+		if (this.isPopulating) {
+			return;
+		}
+		name.replace(`[${itemId}]`, '');
+
+		const stored = this.store.get(itemId);
+		if (stored) {
+			const storedValue = stored.fields?.[name] ?? stored[name];
+			const diff = window.getDifferences.map(storedValue, value);
+
+			if (diff === null) {
+				// Value matches stored — clean up any pending change for this field
+				if (this.changes.has(itemId)) {
+					delete this.changes.get(itemId)[name];
+					// If no real changes left, remove the item entirely
+					const remaining = Object.keys(this.changes.get(itemId))
+						.filter(k => k !== 'id' && k !== 'content');
+					if (remaining.length === 0) {
+						this.changes.delete(itemId);
+						this.changesStore.delete(itemId);
+					}
+				}
+				return;
+			}
+		}
+
+		if (!this.changes.has(itemId)) {
+			this.changes.set(itemId, { id: itemId, content: this.content });
+		}
+		this.changes.get(itemId)[name] = value;
+
+		this.scheduleBackup();
+		if (typeof itemId === 'number' || !String(itemId).includes('group')) {
+			this.scheduleSave();
+		}
+	}
+	scheduleBackup() {
+		window.debouncer.schedule(
+			`changes-${this.content}`,
+			async () => {
+				if (this.changes.size > 0) {
+					await this.handleBackup();
+				}
+			},
+			2000
+		);
+	}
+	cancelBackup() {
+		window.debouncer.cancel(`changes-${this.content}`);
+	}
+	async handleBackup() {
+		const changesArray = Array.from(this.changes.values());
+		this.changes.clear();
+
+		const ids = changesArray.map(c => c.id);
+		const existing = await Promise.all(
+			ids.map(id => this.changesStore.get(id))
+		);
+
+		const changes = changesArray.map((change, i) =>
+			existing[i] ? window.deepMerge(existing[i], change) : change
+		);
+
+		await this.changesStore.saveMany(changes);
+	}
+
+	scheduleSave(delay = 10000) {
+		window.debouncer.schedule(
+			`save-${this.content}`,
+			async () => {
+				// Ensure latest changes are in IndexedDB
+				if (this.changes.size > 0) {
+					this.cancelBackup();
+					await this.handleBackup();
+				}
+
+				await this.savePosts('', false);
+			},
+			delay
+		);
+	}
+	handleFilterChange(target) {
+		let filter = target.dataset.filter;
+
+		if (filter === 'date' && target.value === 'custom') {
+			target.value = '';
+			this.modals.date.handleOpen();
+			return;
+		}
+
+		if (filter === 'date' && target.value !== '') {
+			this.setFilter('date-filter', target.value);
+			// Clear custom range
+			this.deleteFilter('dateFrom');
+			this.deleteFilter('dateTo');
+			this.checkHideFilters();
+			return;
+		}
+
+		if (filter === 'taxonomies') {
+			filter = `tax_${target.dataset.taxonomy}`;
+		}
+
+		this.setFilter(filter, target.value);
+	}
+	checkHideFilters() {
+		const filters = this.store.filters;
+		const hasActiveFilter = Object.entries(filters).some(([key, value]) => {
+			// Skip internal props
+			if (['content', 'user', 'page'].includes(key)) return false;
+			// Check if differs from default
+			return this.defaults[key] !== value && value !== '' && value !== null;
+		});
+
+		this.ui.buttons.clearFilters.hidden = !hasActiveFilter;
+	}
+	clearAllFilters() {
+		let currentFilters = this.store.filters;
+		this.store.clearFilters();
+		for (let [filter, value] of Object.entries(currentFilters)) {
+			this.cache.remove(filter);
+			this.deleteFilter(filter, value);
+		}
+		this.a11y.announce('All filters cleared');
+	}
+
+	handleCustomDateSelection() {
+
+		// Check if month select was used
+		if (this.ui.modals.date.month && this.ui.modals.date.month.value) {
+			const [year, month] = this.ui.modals.date.month.value.split('-');
+			const firstDay = `${year}-${month}-01`;
+			const lastDay = new Date(year, parseInt(month), 0).getDate();
+			const lastDayFormatted = `${year}-${month}-${String(lastDay).padStart(2, '0')}`;
+
+			this.setFilter('dateFrom', firstDay);
+			this.setFilter('dateTo', lastDayFormatted);
+
+			// Clear the regular date-filter
+			this.deleteFilter('date-filter');
+
+			// Reset month select for next time
+			this.ui.modals.date.month.value = '';
+		}
+		// Otherwise check custom range
+		else if (this.ui.modals.date.start && this.ui.modals.date.start.value && this.ui.modals.date.end && this.ui.modals.date.end.value) {
+			this.setFilter('dateFrom', this.ui.modals.date.start.value);
+			this.setFilter('dateTo', this.ui.modals.date.end.value);
+
+			// Clear the regular date-filter
+			this.deleteFilter('date-filter');
+
+			// Reset inputs for next time
+			this.ui.modals.date.start.value = '';
+			this.ui.modals.date.end.value = '';
+		}
+
+		this.checkHideFilters();
+	}
+		handleViewChange(view) {
+			this.view = view.dataset.view;
+			this.cache.set('view', this.view);
+			this.render();
+		}
+
+	handleClick(e) {
+		// Use matches() instead of closest() where possible (faster)
+		if (e.target.matches('.clear-search')) {
+			this.deleteFilter('search', '');
+			return;
+		}
+
+		const actionButton = e.target.closest('[data-action]');
+		if (actionButton) {
+			e.preventDefault();
+			this.handleActionButton(actionButton);
+			return;
+		}
+
+		if (e.target.matches('.apply-date-filter')) {
+			this.handleCustomDateSelection();
+			this.modals.date.handleClose();
+			return;
+		}
+
+		if (e.target.matches(this.selectors.buttons.create) || e.target.closest(this.selectors.buttons.create)) {
+			this.openCreateModal();
+		}
+	}
+		openCreateModal(){
+			this.forms.registerForm(this.ui.modals.create.form,{
+				cache: false,
+			});
+			this.ui.modals.create.modal.dataset.itemId = window.generateID('new');
+
+			this.modals.create.handleOpen();
+		}
+		handleActionButton(button) {
+			const itemID = button.dataset.id;
+
+			switch (button.dataset.action) {
+				case 'edit':
+					this.openEditModal(itemID);
+					break;
 				case 'delete':
-					this.setBulkStatus('delete');
+					if (confirm('Delete this item? This cannot be undone')) {
+						this.updateItem(itemID, 'post_status', 'delete');
+						window.fade(button.closest('.item'), false);
+						this.savePosts(`Permanently deleting ${this.singular}...`).then(()=>{});
+						this.store.delete(itemID);
+					}
+					break;
+				case 'trash':
+					if (this.status === 'trash') {
+						if (confirm('Delete this item? This cannot be undone')) {
+							this.updateItem(itemID, 'post_status', 'delete');
+							window.fade(button.closest('.item'), false);
+							this.savePosts(`Permanently deleting ${this.singular}...`).then(()=>{});
+							this.store.delete(itemID);
+						}
+					} else {
+						this.updateItem(itemID, 'post_status', 'trash');
+						window.fade(button.closest('.item'), false);
+						this.savePosts(`Sending ${this.singular} to trash...`).then(()=>{});
+					}
+					break;
+				case 'bulk-edit':
+					if (this.selected.size > 0) {
+						this.openBulkEditModal();
+					}
+					break;
+				case 'bulk-delete':
+					this.handleBulkDelete();
+					break;
+				case 'refresh':
+					this.store.clearCache();
+					this.store.fetch();
+					break;
+				case 'clear-filters':
+					this.clearAllFilters();
 					break;
 			}
 		}
-		if (window.targetCheck(e, 'select[data-filter]')) {
-			this.handleFilterChange(e);
-		}
-	}
-	handleTableChange(e) {
-		const row = e.target.closest('tr[data-id]');
-		if (!row) return;
-
-		const input = e.target;
-		const postID = parseInt(row.dataset.id);
-		const fieldName = input.closest(['data-field'])?.dataset.field;
-		if (!fieldName) return;
-
-		const item = this.store.get(postID);
-		if (!item) return;
-
-		item.fields[fieldName] = this.getInputValue(input);
-
-		this.store.save(item);
-
-		let post = {};
-		post[postID] = item.fields;
-		this.savePosts(post, `Saving changes to ${this.content}`);
-	}
-	handleTimelineTableChange(e) {
-		const tbody = e.target.closest('tbody[data-id]');
-		if (!tbody) return;
-
-		const input = e.target;
-		const fieldName = input.closest('[data-field]')?.dataset.field;
-
-		if (!fieldName) return;
-
-		const parentID = parseInt(tbody.dataset.id);
-		const timelinePoint = input.closest('tr.timeline-point');
-
-		const item = this.store.get(parentID);
-		if (!item) return;
-
-		const value = this.getInputValue(input);
-
-		// Check if this is a specific point, or a shared value
-		if (timelinePoint) {
-			const imgID = timelinePoint.dataset.imageId;
-			if (!item.fields.timeline) {
-				item.fields.timeline = {};
+		handleBulkDelete() {
+			let isTrash = this.status === 'trash';
+			if (this.selected.size > 0 && confirm(`${isTrash ? 'Permanently delete' : 'Send'} ${this.selected.size} ${this.selected.size === 1 ? this.singular : this.plural}${isTrash ? '' : 'to trash'}?`)) {
+				this.selected.forEach(id => {
+					this.store.delete(id);
+					this.updateItem(id, 'post_status', isTrash ? 'delete' : 'trash');
+				});
+				let title = isTrash
+					? `Permanently deleting ${this.selected.size} ${this.selected.size === 1 ? this.singular : this.plural}`
+					: `Sending ${this.selected.size} ${this.selected.size === 1 ? this.singular : this.plural} to trash`;
+				this.savePosts(title).then(()=>{});
+				this.selectionHandler.clearSelection();
 			}
-			if (!item.fields.timeline[imgID]) {
-				item.fields.timeline[imgID] = {};
-			}
-			item.fields.timeline[imgID][fieldName] = value;
-		} else {
-			item.fields[fieldName] = value;
 		}
 
-		//Update store directly
-		this.store.save(item);
+	handleInput(e) {
+		e.preventDefault();
+		e.stopPropagation();
+		let query = e.target.value.trim();
+		let key = `${this.content}-search`;
 
-		let changes = {};
-		changes[parentID] = item.fields;
-		this.savePosts(changes, 'Updating progress post');
-	}
-	getInputValue(input) {
-		if (input.type === 'checkbox') {
-			return input.checked ? (input.value || '1') : '';
-		}
-		if (input.type === 'radio') {
-			return input.checked ? input.value : null;
-		}
-		return input.value;
-	}
-
-	openTaxonomyModal(taxonomy) {
-		// Check if jvbSelector exists
-		if (!window.jvbSelector) {
-			console.error('TaxonomySelector not initialized');
+		if (query.length === 0) {
+			this.deleteFilter('search', '');
 			return;
 		}
 
-		// Open the selector in filter mode
-		window.jvbSelector.openForFilter(
-			taxonomy,
-			(selectedIds, taxonomy) => this.handleBulkTaxonomy(selectedIds, taxonomy)
+		// Require minimum 2 characters
+		// if (query.length < 2) {
+		// 	return;
+		// }
+
+		window.debouncer.schedule(
+			key,
+			() => {
+				this.a11y.announce(`Searching for "${query}"...`);
+				this.store.setFilters({ search: query, page: 1 });
+			},
+			300
 		);
 	}
-	handleBulkTaxonomy(selectedIds, taxonomy) {
-		// Callback when terms are selected
-		if (selectedIds.length > 0) {
-			selectedIds = selectedIds.join(',');
-			let changes = {};
-			let selected = Array.from(this.viewController.selectedItems);
 
+	handleKeys(e) {
+		if (!this.tabNav) return;
 
-			selected.forEach(sel => {
-				changes[sel] = {
-					content: this.content
-				};
-				changes[sel][taxonomy] = selectedIds;
-			});
+		if (e.key === 'Tab') {
+			e.preventDefault();
 
+			const currentCell = e.target.closest('[data-field]');
+			const currentRow = e.target.closest('tr');
 
-			let title = `Adding ${selected.length} ${this.config.plural??'posts'} to ${selectedIds.length} ${jvbSettings.labels[taxonomy].plural}`;
-			this.viewController.clearSelection();
-			this.savePosts(changes, title);
+			if (!currentCell || !currentRow) return;
+
+			const fieldName = currentCell.dataset.field;
+			const isShift = e.shiftKey;
+
+			// Find next editable row
+			let targetRow = this.findNextEditableRow(currentRow, isShift);
+
+			// If no target row found, wrap around
+			if (!targetRow) {
+				targetRow = this.wrapToRow(currentRow, isShift);
+			}
+
+			if (targetRow) {
+				this.focusFieldInRow(targetRow, fieldName, isShift);
+			}
+		}
+	}
+	findNextEditableRow(currentRow, goBackward = false) {
+		let row = goBackward ? currentRow.previousElementSibling : currentRow.nextElementSibling;
+
+		// For timeline tables, skip non-editable rows
+		while (row && !this.isEditableRow(row)) {
+			row = goBackward ? row.previousElementSibling : row.nextElementSibling;
+		}
+
+		return row;
+	}
+
+	wrapToRow(currentRow, goBackward = false) {
+		if (this.isTimeline) {
+			// For timeline, stay within the same tbody
+			const tbody = currentRow.closest('tbody');
+			if (!tbody) return null;
+
+			const rows = Array.from(tbody.querySelectorAll('tr'))
+				.filter(row => this.isEditableRow(row));
+
+			return goBackward ? rows[rows.length - 1] : rows[0];
+		} else {
+			// For regular tables, use all rows in tbody
+			if (!this.ui.table.body) return null;
+
+			const rows = Array.from(this.ui.table.body.querySelectorAll('tr'))
+				.filter(row => this.isEditableRow(row));
+
+			return goBackward ? rows[rows.length - 1] : rows[0];
+		}
+	}
+	isEditableRow(row) {
+		// Skip thead/tfoot
+		if (row.closest('thead') || row.closest('tfoot')) {
+			return false;
+		}
+
+		// For timeline, check for specific classes
+		if (this.isTimeline) {
+			return row.classList.contains('shared') || row.classList.contains('timeline-point');
+		}
+
+		// For regular tables, check for data-id
+		return !!row.dataset.itemId;
+	}
+
+	focusFieldInRow(row, fieldName, fromAbove = false) {
+		const targetCell = row.querySelector(`[data-field="${fieldName}"]`);
+		if (!targetCell) return;
+
+		const input = this.findFocusableInput(targetCell);
+		if (input) {
+			input.focus();
+
+			// Select text if it's a text input
+			if (input.select && input.type === 'text') {
+				input.select();
+			}
+
+			// Announce for accessibility
+			const direction = fromAbove ? 'next' : 'previous';
+			this.a11y?.announce(`Moved to ${fieldName} in ${direction} row`);
 		}
 	}
 
-	setBulkStatus(status) {
-		if (!['publish', 'draft', 'trash', 'delete'].includes(status)){
-			return;
+	findFocusableInput(cell) {
+		const selectors = [
+			'input:not([type="hidden"]):not([disabled])',
+			'textarea:not([disabled])',
+			'select:not([disabled])',
+			'button:not([disabled])'
+		];
+
+		for (const selector of selectors) {
+			const element = cell.querySelector(selector);
+			if (element) return element;
 		}
 
-		let changes = {};
-		for (let selected of this.viewController.selectedItems) {
-			changes[selected] = {
-				post_status: status,
-				content: this.content
-			};
+		return null;
+	}
+
+
+	/*******************************************************************
+	 MODALS
+	*******************************************************************/
+	openEditModal(itemID) {
+		let item = this.store.get(parseInt(itemID));
+		if (!item) return;
+		this.activeItem = item.id;
+		this.ui.modals.edit.modal.dataset.itemId = itemID;
+		this.ui.modals.edit.modal.dataset.content = this.content;
+		let title;
+		if (Object.hasOwn(item.fields, 'post_title')) {
+			title = item.fields.post_title;
+		} else if (Object.hasOwn(item.fields, 'name')) {
+			title = item.fields.name;
 		}
+		this.ui.modals.edit.h2.textContent = `Editing ${title === '' ? this.singular : title}`;
+		this.ui.modals.edit.form.dataset.formId = `edit-${itemID}`;
+
+
+		this.modals.edit.handleOpen();
+		this.forms.registerForm(this.ui.modals.edit.form, {cache: false,
+			autoUpload: true,});
+
+
+		this.isPopulating = true;
+		this.populate.populate(this.ui.modals.edit.form, item);
+		//For quill/taxonomy selector's async setups
+		requestAnimationFrame(() => {
+			requestAnimationFrame(() => {
+				this.isPopulating = false;
+			});
+		});
+
+	}
+	openBulkEditModal() {
+		window.removeChildren(this.ui.modals.bulkEdit.selected);
+		this.ui.modals.edit.form.reset();
+
+		window.chunkIt(
+			this.selected,
+			(itemId) => {
+				let item = this.store.get(parseInt(itemId));
+				if (!item) return;
+				itemIds.push(item.id);
+
+				return window.jvbTemplates.create('bulkItem', item);
+			},
+			(fragment) => this.ui.modals.bulkEdit.selected.append(fragment)
+		).then(()=>{});
+		let itemIds = Array.from(this.selected).map(id => this.store.get(parseInt(id))).filter(Boolean);
+
+		this.ui.modals.bulkEdit.modal.dataset.itemId = itemIds.join(',');
+
+		if (this.ui.modals.bulkEdit.h2) {
+			this.ui.modals.bulkEdit.h2.textContent = this.selected.size;
+		}
+		this.modals.bulkEdit.handleOpen();
+
+
+		this.forms.registerForm(this.ui.modals.bulkEdit.form, {cache:false});
+		this.isPopulating = true;
+		this.populate.populate(this.ui.modals.edit.form, item);
+		requestAnimationFrame(() => {
+			requestAnimationFrame(() => {
+				this.isPopulating = false;
+			});
+		});
+	}
+
+	/*****************************************************************
+	 FIELD HANDLING
+	*****************************************************************/
+
+	async savePosts(title = '', delay = false) {
+		if (this.changes.size > 0) {
+			this.cancelBackup();
+			await this.handleBackup();
+		}
+		let changes = await this.changesStore.getAll();
+		if (changes.length === 0) return;
+
+		// Filter out false positives
+		changes = this.validateChanges(changes);
+		if (changes.length === 0) return;
+
+		if (title === '') {
+			title = `Saving ${changes.length} ${changes.length === 1 ? this.singular : this.plural}`;
+		}
+
+		let allChanges = {};
+		let remove = [];
+
+		changes.forEach(change => {
+			let itemId = change.id;
+			const { id, ...changeWithoutId } = change;
+			allChanges[itemId] = changeWithoutId;
+
+			if (change.post_status && this.shouldRemoveItemUI(change.post_status)) {
+				remove.push(itemId);
+			}
+		});
+
+		if (remove.length > 0) {
+			this.removeItems(remove);
+		}
+
+		let operation = {
+			endpoint: this.endpoint,
+			headers: {
+				'X-Action-Nonce': window.auth.getNonce('dash'),
+			},
+			data: {
+				posts: allChanges,
+			},
+			delay: delay,
+			popup: `Saving changes`,
+			title: title
+		};
+		this.queue.addToQueue(operation);
+	}
+
+	/**
+	 * Compare pending changes against the store, removing unchanged fields.
+	 * Returns cleaned array (may be empty if nothing actually changed).
+	 */
+	validateChanges(changes) {
+		return changes.reduce((valid, change) => {
+			const { id, content, ...fields } = change;
+			const stored = this.store.get(id);
+
+			if (!stored) {
+				valid.push(change);
+				return valid;
+			}
+
+			const realChanges = { id, content };
+			let hasRealChange = false;
+
+			for (const [name, value] of Object.entries(fields)) {
+				const storedValue = stored.fields?.[name] ?? stored[name];
+				const diff = window.getDifferences.map(storedValue, value);
+
+				if (diff !== null) {
+					realChanges[name] = value;
+					hasRealChange = true;
+				}
+			}
+
+			if (hasRealChange) {
+				valid.push(realChanges);
+			} else {
+				this.changes.delete(id);
+				this.changesStore.delete(id);
+			}
+
+			return valid;
+		}, []);
+	}
+
+
+	setBulkStatus(status) {
+		if (!['publish', 'draft', 'trash', 'delete'].includes(status)) return;
+		let ids = [];
+		this.selected.forEach(itemID => {
+			ids.push(itemID);
+			this.updateItem(itemID, 'post_status', status);
+		});
 		let title;
 		switch (status) {
 			case 'delete':
@@ -554,150 +1450,426 @@
 			default:
 				title = window.uppercaseFirst(status)+'ing';
 		}
-
-		if ((this.status === 'all' && !['publish', 'draft'].includes(status)) || status !== this.status) {
-			let delay = 0;
-			for (let selected of this.viewController.selectedItems) {
-				setTimeout(() => {
-					const element = document.querySelector(`.item[data-id="${selected}"]`);
-					if (element) {
-						window.fade(element, false);
-					}
-				}, delay);
-				delay += 50; // Increment delay for staggered effect
-			}
+		if (this.shouldRemoveItemUI(status)) {
+			this.removeItems(ids);
 		}
-		// Clear selection even if items aren't being removed
-		this.viewController.clearSelection();
+		this.selectionHandler.clearSelection();
 
+		this.savePosts(`${title} ${ids.length} ${ids.length === 1 ? this.singular : this.plural}...`).then(()=>{});
 
-		if (Object.keys(changes).length !== 0) {
-			this.savePosts(changes, `${title} ${this.viewController.selectedItems.size} ${this.plural}...`);
-		}
 	}
-
-	handleFilterChange(e) {
-		let target = e.target;
-		let filter = target.dataset.filter;
-		if (filter === 'taxonomies') {
-			let taxonomy = target.dataset.taxonomy;
-			this.store.setFilter(`tax_${taxonomy}`, target.value);
-		} else {
-			this[target.dataset.filter] = target.value;
-			this.store.setFilter(target.dataset.filter, target.value);
-			if (target.dataset.filter === 'status') {
-				this.updateBulkOptions(target.value);
-			}
+	/***************************************************************
+	 VIEW
+	***************************************************************/
+	render() {
+		const items = this.store.getFiltered();
+		if (items.length === 0) {
+			this.renderEmpty();
+			return;
 		}
+
+		switch (this.view) {
+			case 'grid':
+				this.renderGrid(items);
+				break;
+			case 'table':
+				this.renderTable(items).then(()=>{});
+				break;
+			case 'list':
+				this.renderList(items);
+				break;
+		}
+		this.updateUI();
 	}
-	updateBulkOptions(status = 'all') {
-		if (status === 'trash') {
-			if (this.ui.bulkSelectActions?.querySelector('[value="edit"]')) {
-				window.removeChildren(this.ui.bulkSelectActions);
-				let options = window.getTemplate('trashOptions');
-				options.querySelectorAll('option').forEach((option, index) => {
-					if (index === 0) {
-						option.checked = true;
-					}
-					this.ui.bulkSelectActions.append(option);
+	updateUI() {
+		if (this.ui.bulk.action) {
+			let options = false;
+			let hasEdit = this.ui.bulk.action.querySelector('[value="edit"]');
+			let currentStatus = this.status;
+
+			if (currentStatus === 'trash' && hasEdit) {
+				window.removeChildren(this.ui.bulk.action);
+				options = window.jvbTemplates.create('trashOptions');
+			} else if (currentStatus !== 'trash' && !hasEdit) {
+				window.removeChildren(this.ui.bulk.action);
+				options = window.jvbTemplates.create('notTrashOptions');
+			}
+			if (options) {
+				options.querySelectorAll('option').forEach((option, index)=> {
+					if (index === 0) option.checked = true;
+					this.ui.bulk.action.append(option);
 				});
 			}
+			this.ui.bulk.action.value = '';
+		}
+		if (this.selected.size > 0) {
+			this.selectionHandler.updateSelectionUI();
+		}
+	}
+
+	renderEmpty() {
+		this.toggleTable(false);
+		window.removeChildren(this.ui.grid);
+		const empty = window.jvbTemplates.create('emptyState');
+		if (empty) {
+			this.ui.grid.append(empty);
+			this.a11y.announceItems(0,false,false);
+		}
+	}
+
+	toggleTable(on = true) {
+		if (this.ui.table.selectedColumns) this.ui.table.selectedColumns.hidden = !on;
+
+		if (on && !this.ui.table.form) {
+			let table = window.jvbTemplates.create('contentTable');
+			this.container.append(table);
+			this.ui.table = window.uiFromSelectors(this.selectors.table);
+			this.ui.table.columns = this.container.querySelectorAll(this.selectors.table.columns);
+		}
+
+		if (this.ui.table.form) {
+			this.ui.table.form.hidden = !on;
+			if (!on){
+				this.forms.clearForm(this.ui.table.form.dataset.formId)
+			}
+			if (this.ui.table.body) {
+				window.removeChildren(this.ui.table.body);
+			}
+		}
+		this.keyHandler = this.handleKeys.bind(this);
+		if (on) {
+			document.addEventListener('keydown', this.keyHandler);
 		} else {
-			if (this.ui.bulkSelectActions && !this.ui.bulkSelectActions.querySelector('[value="edit"]')) {
-				window.removeChildren(this.ui.bulkSelectActions);
+			document.removeEventListener('keydown', this.keyHandler);
+		}
 
-				let options = window.getTemplate('notTrashOptions');
-				options.querySelectorAll('option').forEach((option, index) => {
-					this.ui.bulkSelectActions.append(option);
-				});
-			}
-		}
-		if (this.ui.bulkSelectActions) {
-			this.ui.bulkSelectActions.value = '';
-		}
 	}
 
-	populateBulkEdit() {
-		const container = this.modals.bulkEdit.modal.querySelector('form .selected');
-		if (!container) return;
+	renderGrid(items) {
+		window.removeChildren(this.ui.grid);
+		this.toggleTable(false);
 
-		window.removeChildren(container);
-		for (let selected of this.viewController.selectedItems) {
-			let item = this.store.get(selected);
+		this.ui.grid.classList.remove('list-view');
+		this.ui.grid.classList.add('grid-view');
 
-			const img = window.getTemplate('bulkItem');
-			if (!img) return;
-
-			const checkbox = img.querySelector('input[type=checkbox]');
-			const image = img.querySelector('img');
-
-			if (checkbox) {
-				checkbox.id = `bulk_${item.id}`;
-				checkbox.value = item.id;
-				checkbox.checked = true;
-			}
-
-			if (image && item.thumbnail) {
-				image.src = item.thumbnail;
-				image.alt = item.alt || '';
-			}
-
-			container.append(img);
-		}
-		let modal = this.modals.bulkEdit.modal;
-		[
-			modal.querySelector('h2 span').textContent
-		] = [
-			this.viewController.selectedItems.size
-		];
-
-		this.formController.registerForm(this.ui.forms.bulkEdit);
+		window.chunkIt(
+			items,
+			(item) => this.renderGridItem(item),
+			(fragment) => this.ui.grid.append(fragment)
+		).then(()=>{});
 	}
 
-	populateEditForm(itemID) {
-		this.currentItemID = itemID;
+	renderList(items) {
+		window.removeChildren(this.ui.grid);
+		this.toggleTable(false);
 
-		let item = this.store.get(parseInt(itemID));
-		if (item) {
-			this.ui.modals.edit.dataset.itemId = itemID;
-			this.ui.modals.edit.dataset.content = this.content;
-
-			let form = this.ui.modals.edit.querySelector('form');
-			this.ui.modals.edit.querySelector('h2').textContent = `Editing ${item.fields.post_title}`;
-			form.dataset.formId = `edit-${itemID}`;
-			
-			new window.jvbPopulate(form, item);
-
-			this.formController.registerForm(this.ui.forms.edit);
-		}
+		this.ui.grid.classList.remove('grid-view');
+		this.ui.grid.classList.add('list-view');
+		window.chunkIt(
+			items,
+			(item) => this.renderListItem(item),
+			(fragment) => this.ui.grid.append(fragment)
+		).then(()=>{});
 	}
 
-	setupFilters() {
-		// Search
-		const searchInput = document.querySelector('input[type="search"]');
-		if (searchInput) {
-			let searchTimeout;
-			searchInput.addEventListener('input', () => {
-				if (searchInput.value.length > 3) {
-					clearTimeout(searchTimeout);
-					searchTimeout = setTimeout(() => {
-						this.store.setFilter('search', searchInput.value);
-					}, 300);
-				} else if (searchInput.value.length === 0) {
-					this.store.removeFilter('search');
+	async renderTable(items) {
+		this.toggleTable();
+		window.removeChildren(this.ui.grid);
+
+		await window.chunkIt(
+			items,
+			(item) => this.renderTableItem(item),
+			(fragment) => {
+				if (this.ui.table.body) {
+					this.ui.table.body.append(fragment);
+				} else {
+					this.ui.table.table.insertBefore(fragment, this.ui.table.foot);
 				}
-			});
-		}
+			},
+			5
+		);
+
+		requestAnimationFrame(() => {
+			window.jvbSelector?.scanExistingFields(this.ui.table.table);
+		});
 	}
 
-	destroy() {
-		document.querySelectorAll('[data-filter]').forEach(filter => {
-			filter.removeEventListener('change', this.filterHandler);
+	/***************************************************************
+ 	RENDER HELPERS
+	 ***************************************************************/
+	renderGridItem(item) {
+		let gridItem = window.jvbTemplates.create('gridView', item);
+		this.items.set(item.id, gridItem);
+		return gridItem;
+	}
+
+	renderListItem(item) {
+		let listItem = window.jvbTemplates.create('listView', item);
+		this.items.set(item.id, listItem);
+		return listItem;
+	}
+
+	renderTableItem(item) {
+		let tableItem = window.jvbTemplates.create('tableView', item);
+		this.items.set(item.id, tableItem);
+		return tableItem;
+	}
+
+	toggleColumn(column, show) {
+		this.ui.table.table.querySelectorAll(`.${column}`).forEach(el =>{
+			el.hidden = !show;
 		});
 	}
+	/***************************************************************
+	 UPLOAD GROUP SUPPORT
+	 Handles:
+	  	- immediate UI feedback once the uploaded groups are sent to server
+	***************************************************************/
+	handleGroupsUploaded(data) {
+		const { posts, fieldId } = data;
+		let uploader = window.jvbUploads;
+		let field = uploader.fields.get(fieldId);
+
+		let added = [];
+		posts.forEach(post => {
+			const placeholderPost = {
+				id: post.groupId,
+				title: post.fields.post_title || `New ${this.singular}`,
+				status: 'draft',
+				date: new Date().toISOString(),
+				modified: new Date().toISOString(),
+				thumbnail: null,
+				icon: this.content,
+				taxonomies: {},
+				fields: post.fields,
+				images: {},
+			};
+
+			post.images.forEach((uploadId, index) => {
+				let id = uploadId['upload_id'];
+				if (index === 0) {
+					placeholderPost.fields['post_thumbnail'] = uploadId;
+				}
+				let upload = uploader.stores.uploads.get(id);
+				if (upload) {
+					placeholderPost.images[id] = {
+						'image-alt-text': '',
+						'image-caption': '',
+						'image-title': upload.fields.originalName,
+						medium: uploader.createPreviewUrl(uploader.formatFile(upload))
+					};
+				}
+
+			});
+			//
+			// // Add to store (won't persist since it's a fake ID)
+			// this.store.data.set(post.groupId, placeholderPost);
+			//
+			//
+			// // Render immediately
+			// let element;
+			// switch (this.view) {
+			// 	case 'grid':
+			// 		element = this.renderGridItem(placeholderPost);
+			// 		this.ui.grid.prepend(element);
+			// 		break;
+			// 	case 'list':
+			// 		element = this.renderListItem(placeholderPost);
+			// 		this.ui.grid.prepend(element);
+			// 		break;
+			// 	case 'table':
+			// 		element = this.renderTableItem(placeholderPost);
+			// 		if (this.ui.table.body) {
+			// 			this.ui.table.body.prepend(element);
+			// 		}
+			// 		break;
+			// }
+			// element.classList.add('uploading');
+			added.push(placeholderPost);
+		});
+		this.store.saveMany(added).then(() => this.render());
+
+
+		this.a11y.announce(`${posts.length} ${posts.length === 1 ? this.singular : this.plural} created. Waiting for server confirmation...`);
+	}
+
+	handleGroupMappings(mappings) {
+		// mappings = { "group_abc123": 456, "group_def456": 789 }
+
+		for (const [groupId, postId] of Object.entries(mappings)) {
+			// Get any pending changes for this temp item
+			let changes = {};
+			if (this.changes.has(groupId)) {
+				changes = this.changes.get(groupId);
+				this.changes.delete(groupId);
+			}
+			let storedChanges = this.changesStore.get(groupId)??{};
+			if (changes.size > 0 || storedChanges.size > 0) {
+				changes = window.deepMerge(storedChanges, changes);
+				this.changes.set(postId, changes);
+				this.scheduleBackup();
+			}
+		}
+	}
+	/***************************************************************
+	 UTILITY
+	***************************************************************/
+	shouldRemoveItemUI(newStatus) {
+		return (this.status === 'all' && !['publish', 'draft'].includes(newStatus))
+			|| newStatus !== this.store.filters.status;
+	}
+	removeItems(items) {
+		items.forEach(itemId => {
+			if (this.items.has(itemId)) {
+				let item = this.items.get(itemId);
+				if (item) window.fade(item, false);
+			}
+		});
+	}
+
+	setFilters(filters) {
+		for (let [key, value] of Object.entries(filters)) {
+			if (!this.allowedFilters.includes(key)) {
+				delete filters[key];
+				continue;
+			}
+			this.cache.set(key, value);
+
+			let el = this.findFilterEl(key);
+			this.setElValue(el, value);
+		}
+		this.store.setFilters(filters);
+	}
+	setFilter(name, value) {
+		if (!this.allowedFilters.includes(name)) return;
+		this.cache.set(name, value);
+
+		if (name === 'status') this.status = value;
+		if (name === 'orderby') this.orderby = value;
+		if (name === 'order') this.order = value;
+
+		let el = this.findFilterEl(name, value);
+		this.setElValue(el, value);
+		this.store.setFilter(name, value);
+	}
+
+	deleteFilter(name, value) {
+		if (!this.allowedFilters.includes(name)) return;
+		if (Object.hasOwn(this.defaults, name)) {
+			this.setFilter(name, this.defaults[name]);
+			return;
+		}
+		let el = this.findFilterEl(name, value);
+		this.setElValue(el, false);
+		this.cache.remove(name);
+		this.setFilter(name, '');
+	}
+	setElValue(element, value) {
+		if (!element) return;
+		if (!value) {
+			if (['SELECT','TEXTAREA'].includes(element.tagName)) element.value = '';
+			if (['text', 'search'].includes(element.type)) element.value = '';
+			if (element.type === 'radio') element.checked = false;
+			return;
+		}
+
+		if (['SELECT','TEXTAREA'].includes(element.tagName)) element.value = value;
+		if (['text', 'search'].includes(element.type)) element.value = value;
+		if (element.type === 'radio') element.checked = true;
+	}
+	findFilterEl(name, value) {
+		//Handle exceptions first (custom date elements)
+		if (['date-filter', 'dateFrom', 'dateTo'].includes(name)) {
+			switch (name) {
+				case 'date-filter':
+					name = 'month';
+					break;
+				case 'dateFrom':
+					name = 'start';
+					break;
+				case 'dateTo':
+					name = 'end';
+					break;
+			}
+			return this.ui.modals.date[name];
+		}
+		// Handle taxonomy filters
+		if (name.includes('tax_')) {
+			const taxonomy = name.replace('tax_', '');
+			const element = this.ui.filters.taxonomies?.[taxonomy];
+			if (element) {
+				return element;
+			}
+			console.warn('Taxonomy filter element not found:', taxonomy);
+			return null;
+		}
+
+		if (!Object.hasOwn(this.ui.filters, name)) {
+			console.warn('Filter el not found: ', name);
+			return false;
+		}
+
+		let el = this.ui.filters[name];
+		if (typeof el === 'object') {
+			if (!Object.hasOwn(this.ui.filters[name], value)) {
+				return false;
+			}
+			el = this.ui.filters[name][value];
+		}
+		return el;
+	}
+	/***************************************************************
+	 CLEANUP
+	***************************************************************/
+	resetForm(form) {
+		// Clear text inputs, textareas
+		form.querySelectorAll('input[type="hidden"], input[type="text"], input[type="number"], input[type="email"], input[type="url"], textarea').forEach(input => {
+			input.value = '';
+		});
+
+		// Uncheck checkboxes and radios
+		form.querySelectorAll('input[type="checkbox"], input[type="radio"]').forEach(input => {
+			input.checked = false;
+		});
+
+		// Reset selects to first option
+		form.querySelectorAll('select').forEach(select => {
+			select.selectedIndex = 0;
+		});
+
+		// Clear any selected items displays
+		form.querySelectorAll('.selected-items').forEach(container => {
+			window.removeChildren(container);
+		});
+
+		// Clear upload previews
+		form.querySelectorAll('.item-grid.preview').forEach(grid => {
+			window.removeChildren(grid);
+		});
+	}
+	destroy() {
+		window.debouncer.cancel(`changes-${this.content}`);
+		if (this.changes.size > 0) {
+			this.changesStore.saveMany(this.changes).then(()=>{});
+			this.changes.clear();
+		}
+		if (this.timelineSortables) {
+			this.timelineSortables.forEach(sortable => sortable.destroy());
+			this.timelineSortables = [];
+		}
+		for (let [name, modal] of Object.entries(this.ui.modals)) {
+			if (modal.form) {
+				modal.form.removeEventListener('submit', this.submitHandler);
+			}
+		}
+		document.removeEventListener('click', this.clickHandler);
+		document.removeEventListener('change', this.changeHandler);
+		if (this.ui.filters.search) {
+			this.ui.filters.search.removeEventListener('input', this.handleInput);
+		}
+	}
 }
 
-// Initialize when ready
 document.addEventListener('DOMContentLoaded', async function()  {
 	window.auth.subscribe((event) => {
 		if (event === 'auth-loaded') {

--
Gitblit v1.10.0