Closed abbrechen closed 3 years ago
Ok, the problem was behind the keyboard. The main issue was that my js file needs the prefix "resources" because after the bundling the files get a "resources" prefix and the index.html is not part of that. Two other problems you will maybe face when integrating VueJs are
The error message TypeError: this.getOptions is not a function
This issue depends on the sass-loader version you are using. When I installed the sass-loader it was automatically v11. I downgraded to ^10.2.0
and that worked.
VueLoaderPlugin
When I used the vue-loader v15 I got an error. Unfortunately, I can't remember the specific text. But I downgraded my loader to ^14.2.4
and then it was fine.
A secondary issue that is similar to the main problem of this post is when I switch the route, the plugin tries to open a file. The routing itself works, so I will work later on this issue. The error message is: Not allowed to load local resource: file:///routeName
(*routeName = placeholder for the route name).
A secondary issue [...] the plugin tries to open a file. [...] The error message is: Not allowed to load local resource: file:///routeName (*routeName = placeholder for the route name).
I finally found a solution on how to load images in Vue components in the webview.
VueJS itself does not allow to load local files, due to security reasons. I think if you run Vue on a server, you don't have this issue. But the webview in Sketch plugins doesn't start a webserver. So the solution is to use url-loader
instead of file-loader
.
The url-loader
inlines images and other files as base64 (= bytes of the image get encoded as string directly into the document):
<!-- default plugin logo 'icon.png' as local file -->
<img src="./assets/icon.png">
<!-- default plugin logo 'icon.png' as base64-->
<img src="">
Keep in mind that there is a size limit for files handled by url-loader
. If the file exceeds the limit
, the fallback is to use file-loader
(which will break the image loading again). You can adjust this value in webpack.skpm.config.js
(see https://v4.webpack.js.org/loaders/url-loader/#limit).
Also keep in mind that images only seem to get processed in the build step. So make sure to repeatedly run npm run watch
(after adding images, fonts or other local files with src
or inside css url()
).
directory structure: (only relevant files shown)
resources/
assets/
icon.png
App.vue
app.js
index.html
style.css
package.json
webpack.skpm.config.js
package.json
:
{
"devDependencies": {
"@skpm/builder": "^0.8.0",
"@skpm/extract-loader": "^2.0.3",
"css-loader": "^3.6.0",
"html-loader": "^0.5.5",
"node-sass": "^6.0.1",
"sass-loader": "^10.2.1",
"url-loader": "^4.1.1",
"vue-loader": "^15.9.8",
"vue-style-loader": "^4.1.3",
"vue-template-compiler": "^2.6.14"
},
"dependencies": {
"sketch-module-web-view": "^3.5.1",
"vue": "^2.6.14"
}
}
webpack.skpm.config.js
:
// We can use 'entry.isPluginCommand' to check, if it is a normal JavaScript file (not a command).
// based on https://github.com/skpm/with-webview/blob/master/template/webpack.skpm.config.js
const { VueLoaderPlugin } = require('vue-loader');
module.exports = function (config, entry) {
config.node = entry.isPluginCommand ? false : {
setImmediate: false
};
config.module.rules.push({
test: /\.(html)$/,
use: [
{ loader: "@skpm/extract-loader" },
{
loader: "html-loader",
options: {
attrs: ['img:src', 'link:href'],
interpolate: true,
},
},
]
});
config.module.rules.push({
test: /\.(css)$/,
use: [
{ loader: "@skpm/extract-loader" },
{ loader: "css-loader" },
]
});
config.module.rules.push({
// see https://vue-loader.vuejs.org/guide/pre-processors.html#sass
test: /\.scss$/,
use: [
'vue-style-loader',
'css-loader',
'sass-loader'
]
});
config.module.rules.push({
test: /\.(png|jpg|gif|svg|sketch)$/,
loader: 'url-loader',
options: {
name: '[name].[ext]',
esModule: false, // see https://stackoverflow.com/a/59075858/webpack-file-loader-outputs-object-module
},
});
config.module.rules.push({
test: /\.vue$/,
use: "vue-loader",
})
config.resolve = {
extensions: ['.js', '.vue', '.json'],
};
if (config.entry && !entry.isPluginCommand) {
config.plugins.push(
new VueLoaderPlugin()
);
}
}
app.js
:
import Vue from 'vue';
import App from './App.vue';
new Vue({
el: '#app',
render: h => h(App),
});
App.vue
:
<template>
<div id="app">
<img src="./assets/icon.png">
<h1>\{{ msg }}</h1>
</template>
<script>
export default {
name: 'app',
data() {
return {
msg: 'Welcome to Your Vue.js App',
};
},
};
</script>
<style lang="scss">
#app {
width: 240px;
height: 180px;
background-image: url("./assets/icon.png");
}
</style>
index.html
:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Webview with Vue</title>
<link rel="stylesheet" href="./style.css" />
</head>
<body>
<div id="app"></div>
<script src="../resources_app.js"></script>
</body>
</html>
Hi,
when I upgraded the webview template with VueJs, the plugin / Safari is not allowed to load my main.js file. I know there is a working vue-webview but it's outdated (Sketch API syntax, sass-loader compatibility). I don't get any error while compiling, just the webview console error:
Not allowed to load local resource: file:///Users/USERNAME/PATH/newWebviewTest/newwebviewtest.sketchplugin/Contents/Resources/main.js
System: macOS Catalina, Version 10.15.7 Sketch: 71.2
Project Structure
webpack.skpm.config.js
index.html
my-command.js
manifest.json
package.json
If anyone knows why I can not load my main.js file, thanks a lot.