- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在创建 indexedDB 数据库 products_db
。
onload
添加 objectStore products_os
到数据库。products_os
。onclick
的 add 按钮,尝试打开具有更高版本号的数据库并创建一个新的 objectStore cart_os
。 这是它没有按预期工作的地方。当前行为:
Application
选项卡中Developer Tools
indexedDB
中没有 product_db
数据库。onblocked
事件的警报如在function addCartData(e) {
弹出。 (此站点打开时没有其他选项卡!)。关闭弹出窗口后,现在点击相同的添加按钮不会执行任何操作。预期行为:
db
和 objectStore
。onclick
,应使用新版本号打开数据库,并向其中添加 objectStore cart_os
。以便稍后我可以使用 cart_os
进行 read/write
交易。index.html
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>Product Table</title>
<link href="styles.css" rel="stylesheet">
<script src="script.js" defer></script>
</head>
<body>
<table id='idb-table' style="width=100%" />
</body>
</html>
脚本.js
// Create an instance of a db object for us to store the open database in
let db;
window.onload = function() {
// Open our database; it is created if it doesn't already exist
let request = window.indexedDB.open('products_db', 1);
// onerror handler signifies that the database didn't open successfully
request.onerror = function() {
console.log('Database failed to open');
}
// onsuccess handler signifies that the database opened successfully
request.onsuccess = function() {
console.log('Database opened successfully');
// Store the opened database object in the db variable. This is used a lot below
db = request.result;
// Run the fetchProducts() function to fetch data from external API
fetchProducts();
}
// Setup the database tables if this has not already been done
request.onupgradeneeded = function(e) {
// Grab a reference to the opened database
let db = e.target.result;
// Create an objectStore to store our products in (basically like a single table)
// including a auto-incrementing key
let objectStore = db.createObjectStore('products_os', {keyPath: 'id', autoIncrement: true});
// Define what data items the objectStore will contain
objectStore.createIndex('title', 'title', {unique: false});
objectStore.createIndex('price', 'price', {unique: false});
objectStore.createIndex('inStock', 'inStock', {unique: false});
}
function fetchProducts() {
let product1 = fetch('http://localhost:3001/location/pathname') // see product.json file
.then(function(response) {
return response.json();
})
.then(function(response) {
addData(response);
})
}
function addData(value) {
// open a read/write db transaction, ready for adding the data
let transaction = db.transaction(['products_os'], 'readwrite');
// call an object store that's already been added to the database
let objectStore = transaction.objectStore('products_os');
let newItem = {
title: value.title,
price: value.sellingPrice,
inStock: value.inStock,
};
// Make a request to add our newItem object to the object store
var request = objectStore.add(newItem);
transaction.oncomplete = function() {
console.log('Transaction completed: database modification finished.');
// update the display of data to show the newly added item, by running displayData() again.
displayData();
}
transaction.onerror = function() {
console.log('Transaction not opened due to error');
};
}
function displayData() {
// Open our object store and then get a cursor - which iterates through all the
// different data items in the store
let objectStore = db.transaction('products_os').objectStore('products_os');
objectStore.openCursor().onsuccess = function(e) {
// Get a reference to the cursor
let cursor = e.target.result;
// Get a reference to table
let table = document.querySelector('#idb-table');
// If there is still another data item to iterate through, keep running this code
if(cursor) {
// Insert into table,
// structure the HTML fragment, and append it inside the table
let tableRow = document.createElement('tr');
for (let val in cursor.value) {
if (Object.prototype.hasOwnProperty.call(cursor.value, val)) {
if (val !== 'id') {
let td = document.createElement('td');
tableRow.appendChild(td);
td.textContent = cursor.value[val];
}
}
}
let td = document.createElement('td');
td.setAttribute('align', 'center');
let addButton = document.createElement('button');
td.appendChild(addButton);
addButton.textContent = 'Add';
addButton.addEventListener('click', function(e) {
addCartData(e);
})
tableRow.appendChild(td);
table.appendChild(tableRow);
// Iterate to the next item in the cursor
cursor.continue();
}
}
}
function addCartData(e) {
// Open the db with higher version number
let req = window.indexedDB.open('products_db', 2);
req.onerror = function() {
console.log('Database failed to open');
}
req.onsuccess = function() {
console.log('Database opened successfully');
// Store the opened database object in the db variable. This is used a lot below
db = req.result;
}
req.onblocked = function(event) {
// If some other tab is loaded with the database, then it needs to be closed
// before we can proceed.
alert("Please close all other tabs with this site open!");
};
req.onupgradeneeded = function(e){
// Grab a reference to the opened database
let db = e.target.result;
// Create a cart objectStore to store our information added to the cart
let cartStore = db.createObjectStore('cart_os', {keyPath: 'id', autoIncrement: true});
// Define what data items the cart objectStore will contain
cartStore.createIndex('title', 'title', {unique: false});
cartStore.createIndex('price', 'price', {unique: false});
}
}
样式.css
table {
font-family: arial, sans-serif;
border-collapse: collapse;
width: 100%;
}
td, th {
border: 1px solid #dddddd;
text-align: left;
padding: 8px;
}
tr:nth-child(even) {
background-color: #dddddd;
}
tr:last-child {
text-align: center!important;
}
产品.json
{
"title": "Newhide Designer",
"productDescription": "",
"sellingPrice": 119,
"inStock": true,
}
编辑
objectStore products_os
was never added to db
我收到错误 Uncaught (in promise) DOMException: Failed to execute 'transaction on 'IDBDatabase': One of the specified object stores was not found.
at line 207 如图所示。
onupgradeneeded_version1:
function onupgradeneeded_version1(e) {
// Grab a reference to the opened database
let db = e.target.result;
// Create an objectStore to store our products in (basically like a single table)
// including a auto-incrementing key
let objectStore = db.createObjectStore('products_os', {keyPath: 'id', autoIncrement: true});
// Define what data items the objectStore will contain
objectStore.createIndex('title', 'title', {unique: false});
objectStore.createIndex('price', 'price', {unique: false});
objectStore.createIndex('inStock', 'inStock', {unique: false});
}
最佳答案
嗯,关于如何改进这段代码,我有很多建议,但直截了本地说,我认为问题源于这样一个事实,即您最初打开一个数据库连接,并保持它打开(无限期地) ),然后在您的单击处理程序中,打开具有更高版本号的第二个数据库版本,如果单击该版本,则在初始数据库连接仍处于打开状态时发生。
我建议将 indexedDB.open => db
逻辑封装在一个 promise 中。这将使代码更易于阅读和编写。然后我建议改变您使用该 promise 的方式。
这是我的想法:
function open(name, version, onupgradeneeded) {
return new Promise(function executor(resolve, reject) {
const request = indexeDB.open(name, version);
request.onsuccess = function(event) { resolve(request.result); };
request.onerror = function(event) { reject(request.result); };
request.onblocked = function(event) { console.debug('blocked indefinitely'); };
request.onupgradeneeded = onupgradeneeded;
});
}
async function fetch_product() {
const response = await fetch('http://localhost:3001/location/pathname');
const data = await response.json();
return data;
}
function add_product(db, data) {
return new Promise(function(resolve, reject) {
const transaction = db.transaction(..., 'readwrite');
transaction.oncomplete = resolve;
transaction.onerror = reject;
const store = transaction.objectStore(...);
store.put(data);
});
}
function get_products(db) {
return new Promise(function(resolve, reject) {
const transaction = db.transaction(...);
const store = transaction.objectStore(...);
const request = store.getAll();
request.onsuccess = function(event) {
const products_array = request.result;
resolve(products_array);
};
request.onerror = function(event) {
reject(request.error);
};
});
}
function render_product(product) {
const table = document.getElementById('table-id');
// create rows and all that.
// and append
button.onclick = handle_click;
}
async function handle_click(event) {
// Based on what was clicked, find and build the product data object
// event.target is the button
const row = event.target.closest('tr');
const name = row.querySelector('td.namecolumn').textContent;
// etc
const product = {
name: name
};
const db = await open(name, version2, onupgradeneeded_for_version2);
await add_product(db, product);
db.close();
}
async function onload() {
const product = await fetch_product();
const db = await open(name, version, onupgradeneeded_version1);
await add_product(db, product);
const products = await get_products(db);
// we are done with the db now
db.close();
// print out product data
for(const product of products) {
render_product(product);
}
}
关于javascript - 尝试打开具有更高版本号的数据库时触发 onblocked,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55550726/
这个问题已经有答案了: jQuery trigger click vs click ()? (3 个回答) 已关闭 5 年前。 我无法区分 trigger('click')与 trigger('cli
我正在运行 VS 2008 和 .NET 3.5 SP1。 我想在 HttpModule 中实现命中跟踪在我的 ASP.NET 应用程序中。很简单,我想。然而,BeginRequest我的事件 Htt
这是一段代码,我收到以下错误 #1064 - You have an error in your SQL syntax; check the manual that corresponds to yo
有没有办法用任意增量触发滚轮事件。就像 jQuery 对“点击”所做的那样: $('#selector').trigger('click'); 我需要类似的东西,只需一个滚轮即可: $('#selec
我正在尝试在配音数据库中触发时间。我想检查一下在不出现角色的电影配音中不能对角色进行配音。这是PDM: 和CDM 我是SQL的初学者,但我知道表“DUBBES”中应该有一些触发器。我试图做这样的事情,
这个问题已经有答案了: jquery programmatically click on new dom element (3 个回答) 已关闭 6 年前。 我有一个 jQuery 事件定义如下: $
主菜单的点击代码适用于类更改,但不适用于子菜单...当单击食物或鞋子等子菜单项时,它不会触发警报命令...事实上,悬停非常适合子菜单但不是活跃的 HTML
问题非常简单: $('#btn1').click(function(event){ alert( "pageX: " + event.pageX + "\npa
我使用 Spring 的调度程序 (@EnableScheduling) 并具有以下 @Scheduled 方法,该方法每分钟调用一次: @Component public class Schedul
错误 SQL 查询:文档 CREATE TRIGGER `triggers_div` AFTER INSERT ON `produits` FOR EACH ROW BEGIN INSERT INTO
我想在插入另一个表时填充表中的一些列值,并为特定列设置条件。我使用触发器: CREATE TRIGGER inserttrigger AFTER INSERT ON table1 FOR EACH R
我可以在 5.6 MySQL 环境中使用一些关于触发器的指导。我想创建一个触发器,如果发现具有相同速度的电脑的价格较低,则该触发器会停止更新。 架构是产品(制造商、型号、类型)PC(型号、速度、内
背景:我们有一个 completed_flag,默认为 0,当有人完成调查时更新为 1。我想记录这次更新发生的时间戳 在编写了这个触发器/函数以在标志从 0 触发到 1 时更新时间戳后,我怀疑我这样做
数据库中有两个表 KistStatus和 LastKistStatus .后者将保存 KistStatus 的所有“最新”值。 . KistStatus有大约 174.000 条记录,LastKist
我正在开发一个使用 APNS 的 iPhone 应用程序。我很清楚实现 APNS、创 build 备 token 的过程,等等等等……我不知道如何通过 Web 服务从提供商端触发和启动 APNS。任何
我有这个 javascript,当数量更改时会触发 update_cart... jQuery('div.woocommerce').on('change', '.qty', function
当我单击任何按钮时,click 事件不会被触发。艰难的是,我使用 $("div").on("click", "button", function () { 让它工作,但我想看到它使用 .class 工
如何在我的代码中触发 Android onCreateOptionsMenu 函数,即无需用户单击手机上的选项菜单按钮? 最佳答案 Activity.openOptionsMenu(); 就可以了 关
我将表单包装在 中然后我设置 list android:windowSoftInputMode="adjustResize" (默认 react native )。现在,当我用手指触摸事件手动聚焦一
我有一个 Android 编程问题。使用下面的代码我想验证一个字符串匹配。它验证正常,但 LogCat 显示 TextWatcher 方法在每次击键时触发两次,我不明白为什么。我希望每次击键只触发一次
我是一名优秀的程序员,十分优秀!