/* * Copyright 2007-2008 Richard J. Moore * Copyright 2009 Aaron J. Seigo * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU Library General Public License version 2 as * published by the Free Software Foundation * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details * * You should have received a copy of the GNU Library General Public * License along with this program; if not, write to the * Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "simplejavascriptapplet.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "simplebindings/animationgroup.h" #include "simplebindings/dataengine.h" #include "simplebindings/i18n.h" #include "simplebindings/appletinterface.h" #include "simplebindings/bytearrayclass.h" #include "simplebindings/filedialogproxy.h" #include "simplebindings/variant.h" using namespace Plasma; Q_DECLARE_METATYPE(QPainter*) Q_DECLARE_METATYPE(QStyleOptionGraphicsItem*) QScriptValue constructColorClass(QScriptEngine *engine); QScriptValue constructEasingCurveClass(QScriptEngine *engine); QScriptValue constructFontClass(QScriptEngine *engine); QScriptValue constructGraphicsItemClass(QScriptEngine *engine); QScriptValue constructKUrlClass(QScriptEngine *engine); QScriptValue constructLinearLayoutClass(QScriptEngine *engine); QScriptValue constructGridLayoutClass(QScriptEngine *engine); QScriptValue constructAnchorLayoutClass(QScriptEngine *engine); QScriptValue constructPainterClass(QScriptEngine *engine); QScriptValue constructPenClass(QScriptEngine *engine); QScriptValue constructQPixmapClass(QScriptEngine *engine); QScriptValue constructQPointClass(QScriptEngine *engine); QScriptValue constructQRectFClass(QScriptEngine *engine); QScriptValue constructQSizeFClass(QScriptEngine *engine); QScriptValue constructTimerClass(QScriptEngine *engine); void registerSimpleAppletMetaTypes(QScriptEngine *engine); class DummyService : public Service { public: ServiceJob *createJob(const QString &operation, QMap ¶meters) { Q_UNUSED(operation) Q_UNUSED(parameters) return 0; } }; void registerEnums(QScriptEngine *engine, QScriptValue &scriptValue, const QMetaObject &meta) { //manually create enum values. ugh for (int i = 0; i < meta.enumeratorCount(); ++i) { QMetaEnum e = meta.enumerator(i); //kDebug() << e.name(); for (int i=0; i < e.keyCount(); ++i) { //kDebug() << e.key(i) << e.value(i); scriptValue.setProperty(e.key(i), QScriptValue(engine, e.value(i))); } } } KSharedPtr SimpleJavaScriptApplet::s_widgetLoader; QHash SimpleJavaScriptApplet::s_animationDefs; SimpleJavaScriptApplet::SimpleJavaScriptApplet(QObject *parent, const QVariantList &args) : Plasma::AppletScript(parent) { Q_UNUSED(args) // kDebug() << "Script applet launched, args" << applet()->startupArguments(); m_engine = new QScriptEngine(this); } SimpleJavaScriptApplet::~SimpleJavaScriptApplet() { if (s_widgetLoader.count() == 1) { s_widgetLoader.clear(); } } void SimpleJavaScriptApplet::reportError(QScriptEngine *engine, bool fatal) { SimpleJavaScriptApplet *jsApplet = qobject_cast(engine->parent()); const QString failureMsg = i18n("Script failure on line %1:\n%2", QString::number(engine->uncaughtExceptionLineNumber()), engine->uncaughtException().toString()); if (jsApplet) { if (fatal) { jsApplet->setFailedToLaunch(true, failureMsg); } else { jsApplet->showMessage(KIcon("dialog-error"), failureMsg, Plasma::ButtonNone); } } else { kDebug() << failureMsg; } kDebug() << engine->uncaughtExceptionBacktrace(); } void SimpleJavaScriptApplet::configChanged() { QScriptValue fun = m_self.property("configChanged"); if (!fun.isFunction()) { kDebug() << "Script: plasmoid.configChanged is not a function, " << fun.toString(); return; } QScriptContext *ctx = m_engine->pushContext(); ctx->setActivationObject(m_self); //kDebug() << "calling plasmoid"; fun.call(m_self); m_engine->popContext(); if (m_engine->hasUncaughtException()) { reportError(m_engine); } } void SimpleJavaScriptApplet::dataUpdated(const QString &name, const DataEngine::Data &data) { QScriptValue fun = m_self.property("dataUpdated"); if (!fun.isFunction()) { kDebug() << "Script: dataUpdated is not a function, " << fun.toString(); return; } QScriptValueList args; args << m_engine->toScriptValue(name) << m_engine->toScriptValue(data); QScriptContext *ctx = m_engine->pushContext(); ctx->setActivationObject(m_self); fun.call(m_self, args); m_engine->popContext(); if (m_engine->hasUncaughtException()) { reportError(m_engine); } } void SimpleJavaScriptApplet::executeAction(const QString &name) { //callFunction("action_" + name); QScriptValue fun = m_self.property("action_" + name); if (fun.isFunction()) { QScriptContext *ctx = m_engine->pushContext(); ctx->setActivationObject(m_self); fun.call(m_self); m_engine->popContext(); if (m_engine->hasUncaughtException()) { reportError(m_engine); } } } void SimpleJavaScriptApplet::paintInterface(QPainter *p, const QStyleOptionGraphicsItem *option, const QRect &contentsRect) { Q_UNUSED(option) Q_UNUSED(contentsRect) //kDebug() << "paintInterface() (c++)"; QScriptValue fun = m_self.property("paintInterface"); if (!fun.isFunction()) { //kDebug() << "Script: paintInterface is not a function, " << fun.toString(); AppletScript::paintInterface(p, option, contentsRect); return; } QScriptValueList args; args << m_engine->toScriptValue(p); args << m_engine->toScriptValue(const_cast(option)); args << m_engine->toScriptValue(contentsRect); QScriptContext *ctx = m_engine->pushContext(); ctx->setActivationObject(m_self); fun.call(m_self, args); m_engine->popContext(); if (m_engine->hasUncaughtException()) { reportError(m_engine); } } QList SimpleJavaScriptApplet::contextualActions() { return m_interface->contextualActions(); } void SimpleJavaScriptApplet::callFunction(const QString &functionName, const QScriptValueList &args) { QScriptValue fun = m_self.property(functionName); if (fun.isFunction()) { QScriptContext *ctx = m_engine->pushContext(); ctx->setActivationObject(m_self); fun.call(m_self, args); m_engine->popContext(); if (m_engine->hasUncaughtException()) { reportError(m_engine); } } } void SimpleJavaScriptApplet::constraintsEvent(Plasma::Constraints constraints) { QString functionName; if (constraints & Plasma::FormFactorConstraint) { callFunction("formFactorChanged"); } if (constraints & Plasma::LocationConstraint) { callFunction("locationChanged"); } if (constraints & Plasma::ContextConstraint) { callFunction("currentActivityChanged"); } if (constraints & Plasma::SizeConstraint) { callFunction("sizeChanged"); } if (constraints & Plasma::ImmutableConstraint) { callFunction("immutabilityChanged"); } } bool SimpleJavaScriptApplet::include(const QString &path) { QFile file(path); if (!file.open(QIODevice::ReadOnly | QIODevice::Text)) { kWarning() << i18n("Unable to load script file: %1", path); return false; } QString script = file.readAll(); //kDebug() << "Script says" << script; // change the context to the parent context so that the include is actually // executed in the same context as the caller; seems to be what javascript // coders expect :) QScriptContext *ctx = m_engine->currentContext(); if (ctx && ctx->parentContext()) { ctx->setActivationObject(ctx->parentContext()->activationObject()); ctx->setThisObject(ctx->parentContext()->thisObject()); } m_engine->evaluate(script, path); if (m_engine->hasUncaughtException()) { reportError(m_engine, true); return false; } return true; } void SimpleJavaScriptApplet::populateAnimationsHash() { if (s_animationDefs.isEmpty()) { s_animationDefs.insert("fade", Plasma::Animator::FadeAnimation); s_animationDefs.insert("geometry", Plasma::Animator::GeometryAnimation); s_animationDefs.insert("grow", Plasma::Animator::GrowAnimation); s_animationDefs.insert("pulse", Plasma::Animator::PulseAnimation); s_animationDefs.insert("rotate", Plasma::Animator::RotationAnimation); s_animationDefs.insert("rotateStacked", Plasma::Animator::RotationStackedAnimation); s_animationDefs.insert("slide", Plasma::Animator::SlideAnimation); s_animationDefs.insert("zoom", Plasma::Animator::ZoomAnimation); } } bool SimpleJavaScriptApplet::init() { connect(applet(), SIGNAL(extenderItemRestored(Plasma::ExtenderItem*)), this, SLOT(extenderItemRestored(Plasma::ExtenderItem*))); setupObjects(); if (!importExtensions()) { return false; } kDebug() << "ScriptName:" << applet()->name(); kDebug() << "ScriptCategory:" << applet()->category(); return include(mainScript()); } void SimpleJavaScriptApplet::extenderItemRestored(Plasma::ExtenderItem* item) { QScriptValueList args; args << m_engine->newQObject(item, QScriptEngine::AutoOwnership, QScriptEngine::PreferExistingWrapperObject); callFunction("initExtenderItem", args); } QScriptValue SimpleJavaScriptApplet::runApplication(QScriptContext *context, QScriptEngine *engine) { Q_UNUSED(engine) if (context->argumentCount() == 0) { return false; } KUrl::List urls; if (context->argumentCount() > 1) { urls = qscriptvalue_cast(context->argument(1)); } const QString app = context->argument(0).toString(); KService::Ptr service = KService::serviceByStorageId(app); if (service) { return KRun::run(*service, urls, 0); } const QString exec = KGlobal::dirs()->findExe(app); if (!exec.isEmpty()) { return KRun::run(exec, urls, 0); } return false; } QScriptValue SimpleJavaScriptApplet::runCommand(QScriptContext *context, QScriptEngine *engine) { Q_UNUSED(engine) if (context->argumentCount() == 0) { return false; } const QString exec = KGlobal::dirs()->findExe(context->argument(0).toString()); if (!exec.isEmpty()) { QString args; if (context->argumentCount() > 1) { const QStringList argList = qscriptvalue_cast(context->argument(1)); if (!argList.isEmpty()) { args = ' ' + KShell::joinArgs(argList); } } return KRun::runCommand(exec + args, 0); } return false; } QScriptValue SimpleJavaScriptApplet::openUrl(QScriptContext *context, QScriptEngine *engine) { Q_UNUSED(engine) if (context->argumentCount() == 0) { return false; } QScriptValue v = context->argument(0); KUrl url = v.isString() ? KUrl(v.toString()) : qscriptvalue_cast(v); if (url.isValid()) { return KRun::runUrl(url, KMimeType::findByUrl(url)->name(), 0); } return false; } QScriptValue SimpleJavaScriptApplet::getUrl(QScriptContext *context, QScriptEngine *engine) { if (context->argumentCount() == 0) { return engine->undefinedValue(); } QScriptValue v = context->argument(0); KUrl url = v.isString() ? KUrl(v.toString()) : qscriptvalue_cast(v); if (!url.isValid()) { return engine->undefinedValue(); } AppletInterface *interface = extractAppletInterface(engine); AllowedUrls allowed = interface ? interface->allowedUrls() : NoUrls; if (url.isLocalFile()) { if (!(allowed & LocalUrls)) { return engine->undefinedValue(); } } else if (!(allowed & NetworkUrls) && !((allowed & HttpUrls) && (url.protocol() == "http" || url.protocol() == "https"))) { return engine->undefinedValue(); } KIO::Job *job = KIO::get(url); return engine->newQObject(job); } void SimpleJavaScriptApplet::registerGetUrl() { QScriptValue get = m_self.property("getUrl"); if (!get.isValid()) { m_self.setProperty("getUrl", m_engine->newFunction(SimpleJavaScriptApplet::getUrl)); } } bool SimpleJavaScriptApplet::importBuiltinExtesion(const QString &extension) { kDebug() << extension; if ("filedialog" == extension) { FileDialogProxy::registerWithRuntime(m_engine); return true; } else if ("launchapp" == extension) { m_self.setProperty("runApplication", m_engine->newFunction(SimpleJavaScriptApplet::runApplication)); m_self.setProperty("runCommand", m_engine->newFunction(SimpleJavaScriptApplet::runCommand)); m_self.setProperty("openUrl", m_engine->newFunction(SimpleJavaScriptApplet::openUrl)); return true; } else if ("http" == extension) { m_interface->setAllowedUrls(m_interface->allowedUrls() | HttpUrls); registerGetUrl(); return true; } else if ("networkio" == extension) { m_interface->setAllowedUrls(m_interface->allowedUrls() | HttpUrls | NetworkUrls); registerGetUrl(); return true; } else if ("localio" == extension) { m_interface->setAllowedUrls(m_interface->allowedUrls() | LocalUrls); registerGetUrl(); return true; } return false; } bool SimpleJavaScriptApplet::importExtensions() { KPluginInfo info = description(); QStringList requiredExtensions = info.service()->property("X-Plasma-RequiredExtensions", QVariant::StringList).toStringList(); kDebug() << "required extensions are" << requiredExtensions; foreach (const QString &ext, requiredExtensions) { QString extension = ext.toLower(); if (m_extensions.contains(extension)) { continue; } if (!applet()->hasAuthorization(extension)) { setFailedToLaunch(true, i18n("Authorization for required extension '%1' was denied.", extension)); return false; } if (!importBuiltinExtesion(extension)) { m_engine->importExtension(extension); } if (m_engine->hasUncaughtException()) { reportError(m_engine, true); return false; } else { m_extensions << extension; } } QStringList optionalExtensions = info.service()->property("X-Plasma-OptionalExtensions", QVariant::StringList).toStringList(); kDebug() << "optional extensions are" << optionalExtensions; foreach (const QString &ext, optionalExtensions) { QString extension = ext.toLower(); if (m_extensions.contains(extension)) { continue; } if (!applet()->hasAuthorization(extension)) { continue; } if (!importBuiltinExtesion(extension)) { m_engine->importExtension(extension); } if (m_engine->hasUncaughtException()) { reportError(m_engine); } else { m_extensions << extension; } } return true; } void SimpleJavaScriptApplet::setupObjects() { QScriptValue global = m_engine->globalObject(); // Bindings for animations global.setProperty("animation", m_engine->newFunction(SimpleJavaScriptApplet::animation)); global.setProperty("AnimationGroup", m_engine->newFunction(SimpleJavaScriptApplet::animationGroup)); global.setProperty("ParallelAnimationGroup", m_engine->newFunction(SimpleJavaScriptApplet::parallelAnimationGroup)); // Bindings for data engine m_engine->setDefaultPrototype(qMetaTypeId(), m_engine->newQObject(new DataEngine(), QScriptEngine::ScriptOwnership)); m_engine->setDefaultPrototype(qMetaTypeId(), m_engine->newQObject(new DummyService(), QScriptEngine::ScriptOwnership)); m_engine->setDefaultPrototype(qMetaTypeId(), m_engine->newQObject(new ServiceJob(QString(), QString(), QMap()), QScriptEngine::ScriptOwnership )); bindI18N(m_engine); global.setProperty("dataEngine", m_engine->newFunction(SimpleJavaScriptApplet::dataEngine)); global.setProperty("service", m_engine->newFunction(SimpleJavaScriptApplet::service)); // Expose applet interface const bool isPopupApplet = qobject_cast(applet()); m_interface = isPopupApplet ? new PopupAppletInterface(this) : new AppletInterface(this); m_self = m_engine->newQObject(m_interface); m_self.setScope(global); global.setProperty("plasmoid", m_self); QScriptValue args = m_engine->newArray(); int i = 0; foreach (const QVariant &arg, applet()->startupArguments()) { args.setProperty(i, ::variantToScriptValue(m_engine, arg)); ++i; } global.setProperty("startupArguments", args); registerEnums(m_engine, global, AppletInterface::staticMetaObject); // Add a global loadui method for ui files QScriptValue fun = m_engine->newFunction(SimpleJavaScriptApplet::loadui); global.setProperty("loadui", fun); fun = m_engine->newFunction(SimpleJavaScriptApplet::print); global.setProperty("print", fun); // Work around bug in 4.3.0 qMetaTypeId(); // Add stuff from Qt global.setProperty("QPainter", constructPainterClass(m_engine)); global.setProperty("QGraphicsItem", constructGraphicsItemClass(m_engine)); global.setProperty("QTimer", constructTimerClass(m_engine)); global.setProperty("QFont", constructFontClass(m_engine)); global.setProperty("QColor", constructColorClass(m_engine)); global.setProperty("QEasingCurve", constructEasingCurveClass(m_engine)); global.setProperty("QRectF", constructQRectFClass(m_engine)); global.setProperty("QPen", constructPenClass(m_engine)); global.setProperty("QPixmap", constructQPixmapClass(m_engine)); global.setProperty("QSizeF", constructQSizeFClass(m_engine)); global.setProperty("QPoint", constructQPointClass(m_engine)); global.setProperty("LinearLayout", constructLinearLayoutClass(m_engine)); global.setProperty("GridLayout", constructGridLayoutClass(m_engine)); global.setProperty("AnchorLayout", constructAnchorLayoutClass(m_engine)); ByteArrayClass *baClass = new ByteArrayClass(m_engine); global.setProperty("ByteArray", baClass->constructor()); // Add stuff from KDE libs qScriptRegisterSequenceMetaType(m_engine); global.setProperty("Url", constructKUrlClass(m_engine)); // Add stuff from Plasma global.setProperty("PlasmaSvg", m_engine->newFunction(SimpleJavaScriptApplet::newPlasmaSvg)); global.setProperty("PlasmaFrameSvg", m_engine->newFunction(SimpleJavaScriptApplet::newPlasmaFrameSvg)); global.setProperty("Svg", m_engine->newFunction(SimpleJavaScriptApplet::newPlasmaSvg)); global.setProperty("FrameSvg", m_engine->newFunction(SimpleJavaScriptApplet::newPlasmaFrameSvg)); global.setProperty("ExtenderItem", m_engine->newFunction(SimpleJavaScriptApplet::newPlasmaExtenderItem)); registerSimpleAppletMetaTypes(m_engine); installWidgets(m_engine); } QSet SimpleJavaScriptApplet::loadedExtensions() const { return m_extensions; } AppletInterface *SimpleJavaScriptApplet::extractAppletInterface(QScriptEngine *engine) { QScriptValue appletValue = engine->globalObject().property("plasmoid"); return qobject_cast(appletValue.toQObject()); } QScriptValue SimpleJavaScriptApplet::dataEngine(QScriptContext *context, QScriptEngine *engine) { if (context->argumentCount() != 1) { return context->throwError(i18n("dataEngine() takes one argument")); } AppletInterface *interface = extractAppletInterface(engine); if (!interface) { return context->throwError(i18n("Could not extract the Applet")); } const QString dataEngine = context->argument(0).toString(); DataEngine *data = interface->dataEngine(dataEngine); return engine->newQObject(data); } QScriptValue SimpleJavaScriptApplet::service(QScriptContext *context, QScriptEngine *engine) { if (context->argumentCount() != 2) { return context->throwError(i18n("service() takes two arguments")); } QString dataEngine = context->argument(0).toString(); AppletInterface *interface = extractAppletInterface(engine); if (!interface) { return context->throwError(i18n("Could not extract the Applet")); } DataEngine *data = interface->dataEngine(dataEngine); QString source = context->argument(1).toString(); Service *service = data->serviceForSource(source); //kDebug( )<< "lets try to get" << source << "from" << dataEngine; return engine->newQObject(service, QScriptEngine::AutoOwnership); } QScriptValue SimpleJavaScriptApplet::animation(QScriptContext *context, QScriptEngine *engine) { if (context->argumentCount() != 1) { return context->throwError(i18n("animation() takes one argument")); } populateAnimationsHash(); QString animName = context->argument(0).toString().toLower(); const bool isPause = animName == "pause"; const bool isProperty = animName == "property"; if (!isPause && !isProperty && !s_animationDefs.contains(animName)) { return context->throwError(i18n("%1 is not a known animation type", animName)); } bool parentIsApplet = false; QGraphicsWidget *parent = extractParent(context, engine, 0, &parentIsApplet); if (isPause) { QPauseAnimation *pause = new QPauseAnimation(parent); return engine->newQObject(pause); } else if (isProperty) { QPropertyAnimation *propertyAnim = new QPropertyAnimation(parent); return engine->newQObject(propertyAnim); } else { Plasma::Animation *anim = Plasma::Animator::create(s_animationDefs.value(animName), parent); if (anim) { if (!parentIsApplet) { anim->setTargetWidget(parent); } return engine->newQObject(anim); } } return context->throwError(i18n("%1 is not a known animation type", animName)); } QScriptValue SimpleJavaScriptApplet::animationGroup(QScriptContext *context, QScriptEngine *engine) { QGraphicsWidget *parent = extractParent(context, engine); SequentialAnimationGroup *group = new SequentialAnimationGroup(parent); return engine->newQObject(group); } QScriptValue SimpleJavaScriptApplet::parallelAnimationGroup(QScriptContext *context, QScriptEngine *engine) { QGraphicsWidget *parent = extractParent(context, engine); ParallelAnimationGroup *group = new ParallelAnimationGroup(parent); return engine->newQObject(group); } QScriptValue SimpleJavaScriptApplet::loadui(QScriptContext *context, QScriptEngine *engine) { if (context->argumentCount() != 1) { return context->throwError(i18n("loadui() takes one argument")); } QString filename = context->argument(0).toString(); QFile f(filename); if (!f.open(QIODevice::ReadOnly)) { return context->throwError(i18n("Unable to open '%1'",filename)); } QUiLoader loader; QWidget *w = loader.load(&f); f.close(); return engine->newQObject(w, QScriptEngine::AutoOwnership); } QString SimpleJavaScriptApplet::findImageFile(QScriptEngine *engine, const QString &file) { AppletInterface *interface = extractAppletInterface(engine); if (!interface) { return QString(); } return interface->package()->filePath("images", file); } QString SimpleJavaScriptApplet::findSvg(QScriptEngine *engine, const QString &file) { AppletInterface *interface = extractAppletInterface(engine); if (!interface) { return file; } QString path = interface->package()->filePath("images", file + ".svg"); if (path.isEmpty()) { path = interface->package()->filePath("images", file + ".svgz"); if (path.isEmpty()) { return file; } } return path; } QScriptValue SimpleJavaScriptApplet::newPlasmaSvg(QScriptContext *context, QScriptEngine *engine) { if (context->argumentCount() == 0) { return context->throwError(i18n("Constructor takes at least 1 argument")); } const QString filename = context->argument(0).toString(); bool parentedToApplet = false; QGraphicsWidget *parent = extractParent(context, engine, 1, &parentedToApplet); Svg *svg = new Svg(parent); svg->setImagePath(parentedToApplet ? findSvg(engine, filename) : filename); QScriptValue fun = engine->newQObject(svg); registerEnums(engine, fun, *svg->metaObject()); return fun; } QScriptValue SimpleJavaScriptApplet::newPlasmaFrameSvg(QScriptContext *context, QScriptEngine *engine) { if (context->argumentCount() == 0) { return context->throwError(i18n("Constructor takes at least 1 argument")); } QString filename = context->argument(0).toString(); bool parentedToApplet = false; QGraphicsWidget *parent = extractParent(context, engine, 1, &parentedToApplet); FrameSvg *frameSvg = new FrameSvg(parent); frameSvg->setImagePath(parentedToApplet ? filename : findSvg(engine, filename)); QScriptValue fun = engine->newQObject(frameSvg); // FIXME: why is this necessary when it is clearly declared in FrameSvg's moc? registerEnums(engine, fun, *frameSvg->metaObject()); return fun; } QScriptValue SimpleJavaScriptApplet::newPlasmaExtenderItem(QScriptContext *context, QScriptEngine *engine) { Plasma::Extender *extender = 0; if (context->argumentCount() > 0) { extender = qobject_cast(context->argument(0).toQObject()); } if (!extender) { AppletInterface *interface = extractAppletInterface(engine); if (!interface) { engine->undefinedValue(); } extender = interface->extender(); } Plasma::ExtenderItem *extenderItem = new Plasma::ExtenderItem(extender); QScriptValue fun = engine->newQObject(extenderItem); registerEnums(engine, fun, *extenderItem->metaObject()); return fun; } QScriptValue SimpleJavaScriptApplet::widgetAdjustSize(QScriptContext *context, QScriptEngine *engine) { QGraphicsWidget *widget = qobject_cast(context->thisObject().toQObject()); if (widget) { widget->adjustSize(); } return engine->undefinedValue(); } void SimpleJavaScriptApplet::installWidgets(QScriptEngine *engine) { QScriptValue globalObject = engine->globalObject(); if (!s_widgetLoader) { s_widgetLoader = new UiLoader; } foreach (const QString &widget, s_widgetLoader->availableWidgets()) { QScriptValue fun = engine->newFunction(createWidget); QScriptValue name = engine->toScriptValue(widget); fun.setProperty(QString("functionName"), name, QScriptValue::ReadOnly | QScriptValue::Undeletable | QScriptValue::SkipInEnumeration); fun.setProperty(QString("prototype"), engine->newObject()); globalObject.setProperty(widget, fun); } } QGraphicsWidget *SimpleJavaScriptApplet::extractParent(QScriptContext *context, QScriptEngine *engine, int argIndex, bool *parentedToApplet) { if (parentedToApplet) { *parentedToApplet = false; } QGraphicsWidget *parent = 0; if (context->argumentCount() >= argIndex) { parent = qobject_cast(context->argument(argIndex).toQObject()); } if (!parent) { AppletInterface *interface = extractAppletInterface(engine); if (!interface) { return 0; } //kDebug() << "got the applet!"; parent = interface->applet(); if (parentedToApplet) { *parentedToApplet = true; } } return parent; } QScriptValue SimpleJavaScriptApplet::createWidget(QScriptContext *context, QScriptEngine *engine) { QGraphicsWidget *parent = extractParent(context, engine); QString self = context->callee().property("functionName").toString(); if (!s_widgetLoader) { s_widgetLoader = new UiLoader; } QGraphicsWidget *w = s_widgetLoader->createWidget(self, parent); if (!w) { return QScriptValue(); } QScriptValue fun = engine->newQObject(w); fun.setPrototype(context->callee().property("prototype")); fun.setProperty("adjustSize", engine->newFunction(widgetAdjustSize)); //register enums will be accessed for instance as frame.Sunken for Frame shadow... registerEnums(engine, fun, *w->metaObject()); return fun; } QScriptValue SimpleJavaScriptApplet::notSupported(QScriptContext *context, QScriptEngine *engine) { Q_UNUSED(engine) QString message = context->callee().property("message").toString(); return context->throwError(i18n("This operation was not supported, %1", message) ); } QScriptValue SimpleJavaScriptApplet::print(QScriptContext *context, QScriptEngine *engine) { if (context->argumentCount() != 1) { return context->throwError(i18n("print() takes one argument")); } //TODO: a GUI console? :) std::cout << context->argument(0).toString().toLocal8Bit().constData() << std::endl; return engine->undefinedValue(); } void SimpleJavaScriptApplet::collectGarbage() { m_engine->collectGarbage(); } /* * Workaround the fact that QtScripts handling of variants seems a bit broken. */ QScriptValue SimpleJavaScriptApplet::variantToScriptValue(QVariant var) { return ::variantToScriptValue(m_engine, var); } K_EXPORT_PLASMA_APPLETSCRIPTENGINE(qscriptapplet, SimpleJavaScriptApplet) #include "simplejavascriptapplet.moc"