是否有任何解决方法可以在http和https上使用html5 localstorage?

dan*_*els 13 html5 local-storage

我需要存储一些数据客户端,这些数据太大,无法存储在cookie中.LocalStorage似乎是这样做的完美方式,但事实是,我将使用它的网站有一些部分可以在https和其他只有http和本地存储无法访问您使用http设置的https数据这似乎不再是一个可行的解决方案.

知道是否有任何解决方案吗?还有其他选择吗?

Rob*_*b W 21

将所有数据存储在一个域中,例如https://my.domain.org/.

  • https协议上,只需使用localStorage.setItem('key', 'value')保存数据.
  • http协议上,嵌入一个https框架,并用于postMessage保存数据:

演示:http://jsfiddle.net/gK7ce/4/(辅助页面位于http://jsfiddle.net/gK7ce/3/).

// Script at https://my.domain.org/postMessage
window.addEventListener('message', function(event) {
    // Domain restriction (to not leak variables to any page..)
    if (event.origin == 'http://my.domain.org' ||
        event.origin == 'https://my.domain.org') {
        var data = JSON.parse(event.data);
        if ('setItem' in data) {
            localStorage.setItem(data.setItem, data.value);
        } else if ('getItem' in data) {
            var gotItem = localStorage.getItem(data.getItem);
            // See below
            event.source.postMessage(
                '#localStorage#' + data.identifier + 
                (gotItem === null ? 'null#' : '#' + gotItem),
                event.origin
            );
        } else if ('removeItem' in data) {
            localStorage.removeItem(data.removeItem);
        }
    }
}, false);
Run Code Online (Sandbox Code Playgroud)

在HTTP(S)页面,框架可以如下嵌入(代替https://my.mydomain.com实际的URL请注意,您可以简单地获取到框架的引用和使用.src属性):

<iframe name="myPostMessage" src="https://my.domain.org/postMessage" style="display:none;"></iframe>
// Example: Set the data
function LSsetItem(key, value) {
    var obj = {
        setItem: key,
        value: value
    };
    frames['myPostMessage'].postMessage(JSON.stringify(obj), 'https://my.domain.com');
}
LSsetItem('key', 'value');
Run Code Online (Sandbox Code Playgroud)

请注意,该方法是异步的,因为postMessage.getItem必须以不同方式实现该方法的实现:

var callbacks = {};
window.addEventListener('message', function(event) {
    if (event.source === frames['myPostMessage']) {
        var data = /^#localStorage#(\d+)(null)?#([\S\s]*)/.exec(event.data);
        if (data) {
            if (callbacks[data[1]]) {
                // null and "null" are distinguished by our pattern
                callbacks[data[1]](data[2] === 'null' ? null : data[3]);
            }
            delete callbacks[data[1]];
        }
    }
}, false);
function LSgetItem(key, callback) {
    var identifier = new Date().getTime();
    var obj = {
        identifier: identifier,
        getItem: key
    };
    callbacks[identifier] = callback;
    frames['myPostMessage'].postMessage(JSON.stringify(obj), 'https://my.domain.com');
}
// Usage:
LSgetItem('key', function(value) {
    console.log('Value: ' + value);
});
Run Code Online (Sandbox Code Playgroud)

请注意,每个回调都存储在哈希中.每条消息还包含一个标识符,以便接收消息的窗口调用正确的相应回调.

为了完整性,这是LSremoveItem方法:

function LSremoveItem(key) {
    var obj = {
        removeItem: key
    };
    frames['myPostMessage'].postMessage(JSON.stringify(obj), 'https://my.domain.com');
}
Run Code Online (Sandbox Code Playgroud)

  • @robC Safari的问题是由"阻止cookie和其他网站数据"偏好引起的.当它设置为"来自第三方和广告商"时,不再共享localStorage.它必须设置为"从不",如下图所示:http://i.stack.imgur.com/AP4ed.png. (2认同)