最简单的一个例子是:
bool ScriptablePluginObject::HasMethod(NPIdentifier name){return true;}
bool ScriptablePluginObject::HasProperty(NPIdentifier name){return true;}
bool ScriptablePluginObject::GetProperty(NPIdentifier name, NPVariant *result)
{return NPN_GetProperty(mNpp,sWindowObj,name,result);}
bool ScriptablePluginObject::SetProperty(NPIdentifier name, const NPVariant *value)
{return NPN_SetProperty(mNpp,sWindowObj,name,value);}
bool ScriptablePluginObject::Invoke(NPIdentifier name, const NPVariant *args, uint32_t argCount, NPVariant *result)
{MessageBox(NULL,"functions called","msg",0);return true;}
这样在JS中用任意的字符串代替前面的foo或者bar都会正常的调用前述相关代码。有一点需要指出的是在JS中任何Object都可以设置属性如:
var t=new Object();
t.anystr=val;
alert(t.anystr);
这种方式与前面的不同点在于,这种方式只是JS内核的操作,其设置的属性在插件代码中是无法访问的,插件实现属性设置的机制的目的也就是为了能够方便的访问通过JS设置的属性吧!
要设置几个特定的属性或者方法就需要结合NPN_GetStringIdentifier来实现了。具体见例子程序。
在scriptable的插件中,在Plugin类和ScriptableObject类之间共享数据是非常必要的,使用全局变量当然是一种办法,但是这样会在一个页面中有多个插件实例时产生严重的后果。因此不提倡这样做。注意ScriptableObject的成员mNpp是getScriptObject返回的,而这个mNpp就是Plugin类中的NPP结构,而这个NPP结构的pdata指向的就是Plugin类本身( 在NPP_New中有instance->pdata = (void *)plugin;)。将mNpp->pdata强制转换为Plugin类的指针,然后就可以访问其中的公有成员了。
设置或者获取属性一般也不需要调用NPN的相关接口了,只需要在Plugin中定义一个公有成员变量,在SetProperty中Plugin* plugin=(Plugin*)mNpp->pdata;
plugin->m_bar = NPVARIANT_TO_INT32(*value);在GetProperty中使用Plugin* plugin=(Plugin*)mNpp->pdata;INT32_TO_NPVARIANT(plugin->m_bar,*result);就可以了。