正如我所看到的,Gj imports
默认只加载/usr/share/gjs-1.0
和/usr/lib/gjs-1.0
。我想模块化一个应用程序,就像我们可以用节点做的那样,但我必须找到相对于脚本文件的模块。
我发现了这两种添加包含路径的方法:
gjs --include-path=my-modules my-script.js
GJS_PATH=my-modules gjs my-script.js
...但两者都与当前目录相关,而不是与文件(不经意地)相关,并且需要在命令行上声明它们,这使得这不必要地复杂。
如何在Gjs代码中设置包含路径? (所以我可以将其与文件相关)
或者......还有另一种从任何地方导入文件的方法,比如python?
(请你,你不需要建议使用一个shellcript启动器来解决--include-path
和GJS_PATH
问题。这很明显,但功能较弱。如果我们没有一个更好的解决方案,我们生存下来。)
答案 0 :(得分:10)
您需要设置或修改imports.searchPath
(这并不明显,因为它不会显示for (x in imports)print(x)
)。所以这个:
imports.searchPath.unshift('.');
var foo = imports.foo;
将文件“foo.js”导入为foo
对象。
这与Seed兼容,但imports
知道它有searchPath
。
(这个答案的早期版本实际上不太准确,更具煽动性。抱歉)。
答案 1 :(得分:6)
正如Douglas所说,您需要修改imports.searchPath
以包含您的图书馆位置。使用.
很简单,但取决于始终从同一目录位置运行的文件。不幸的是,找到当前正在执行的脚本的目录是一个巨大的黑客。这是Gnome Shell does it for the extensions API
我已将其改编为以下功能以供一般使用:
const Gio = imports.gi.Gio;
function getCurrentFile() {
let stack = (new Error()).stack;
// Assuming we're importing this directly from an extension (and we shouldn't
// ever not be), its UUID should be directly in the path here.
let stackLine = stack.split('\n')[1];
if (!stackLine)
throw new Error('Could not find current file');
// The stack line is like:
// init([object Object])@/home/user/data/gnome-shell/extensions/u@u.id/prefs.js:8
//
// In the case that we're importing from
// module scope, the first field is blank:
// @/home/user/data/gnome-shell/extensions/u@u.id/prefs.js:8
let match = new RegExp('@(.+):\\d+').exec(stackLine);
if (!match)
throw new Error('Could not find current file');
let path = match[1];
let file = Gio.File.new_for_path(path);
return [file.get_path(), file.get_parent().get_path(), file.get_basename()];
}
在定义app.js
函数后,您可以在输入点文件getCurrentFile
中使用它:
let file_info = getCurrentFile();
// define library location relative to entry point file
const LIB_PATH = file_info[1] + '/lib';
// then add it to the imports search path
imports.searchPath.unshift(LIB_PATH);
嫣!现在导入我们的库是非常容易的:
// import your app libraries (if they were in lib/app_name)
const Core = imports.app_name.core;