附加链接【AHK与其它语法对比】:https://rosettacode.org/wiki/Category:AutoHotkey

这是V1_H版的部分汉化帮助文档:https://www.ahk66.com/ahkh_docs/AutoHotkey.htm

 

虚荣写的C#调用ahkdll的示例:

蓝奏云:https://wwp.lanzouj.com/iVhF41ws6z9c   提取码:ahk6

123网盘:https://www.123pan.com/s/ufi3Td-h0O53.html   提取码:ahk6

 

; https://www.autohotkey.com/boards/viewtopic.php?f=6&t=4633
c# =
(
    using System.Windows.Forms;
    class Foo {
        public void Test() {
            MessageBox.Show("Hello, world, from C#!");
        }
    }
)
asm := CLR_CompileC#(c#, "System.dll | System.Windows.Forms.dll")
obj := CLR_CreateObject(asm, "Foo")
obj.Test()
Return

; C# 与您的 ahk 脚本进行通信的简单方法
; https://www.autohotkey.com/boards/viewtopic.php?p=434643#p434643


; ==========================================================
;                  .NET Framework Interop
;      http://www.autohotkey.com/forum/topic26191.html
; ==========================================================
;
;   Author:     Lexikos
;   Version:    1.2
;   Requires:	AutoHotkey_L v1.0.96+
;

CLR_LoadLibrary(AssemblyName, AppDomain=0)
{
  if !AppDomain
    AppDomain := CLR_GetDefaultDomain()
  e := ComObjError(0)
  Loop 1 {
    if assembly := AppDomain.Load_2(AssemblyName)
      break
    static null := ComObject(13,0)
    args := ComObjArray(0xC, 1),  args[0] := AssemblyName
    typeofAssembly := AppDomain.GetType().Assembly.GetType()
    if assembly := typeofAssembly.InvokeMember_3("LoadWithPartialName", 0x158, null, null, args)
      break
    if assembly := typeofAssembly.InvokeMember_3("LoadFrom", 0x158, null, null, args)
      break
  }
  ComObjError(e)
  return assembly
}

CLR_CreateObject(Assembly, TypeName, Args*)
{
  if !(argCount := Args.MaxIndex())
    return Assembly.CreateInstance_2(TypeName, true)
  
  vargs := ComObjArray(0xC, argCount)
  Loop % argCount
    vargs[A_Index-1] := Args[A_Index]
  
  static Array_Empty := ComObjArray(0xC,0), null := ComObject(13,0)
  
  return Assembly.CreateInstance_3(TypeName, true, 0, null, vargs, null, Array_Empty)
}

CLR_CompileC#(Code, References="", AppDomain=0, FileName="", CompilerOptions="")
{
  return CLR_CompileAssembly(Code, References, "System", "Microsoft.CSharp.CSharpCodeProvider", AppDomain, FileName, CompilerOptions)
}

CLR_CompileVB(Code, References="", AppDomain=0, FileName="", CompilerOptions="")
{
  return CLR_CompileAssembly(Code, References, "System", "Microsoft.VisualBasic.VBCodeProvider", AppDomain, FileName, CompilerOptions)
}

CLR_StartDomain(ByRef AppDomain, BaseDirectory="")
{
  static null := ComObject(13,0)
  args := ComObjArray(0xC, 5), args[0] := "", args[2] := BaseDirectory, args[4] := ComObject(0xB,false)
  AppDomain := CLR_GetDefaultDomain().GetType().InvokeMember_3("CreateDomain", 0x158, null, null, args)
  return A_LastError >= 0
}

CLR_StopDomain(ByRef AppDomain)
{	; ICorRuntimeHost::UnloadDomain
  DllCall("SetLastError", "uint", hr := DllCall(NumGet(NumGet(0+RtHst:=CLR_Start())+20*A_PtrSize), "ptr", RtHst, "ptr", ComObjValue(AppDomain))), AppDomain := ""
  return hr >= 0
}

; NOTE: IT IS NOT NECESSARY TO CALL THIS FUNCTION unless you need to load a specific version.
CLR_Start(Version="") ; returns ICorRuntimeHost*
{
  static RtHst := 0
  ; The simple method gives no control over versioning, and seems to load .NET v2 even when v4 is present:
  ; return RtHst ? RtHst : (RtHst:=COM_CreateObject("CLRMetaData.CorRuntimeHost","{CB2F6722-AB3A-11D2-9C40-00C04FA30A3E}"), DllCall(NumGet(NumGet(RtHst+0)+40),"uint",RtHst))
  if RtHst
    return RtHst
  EnvGet SystemRoot, SystemRoot
  if Version =
    Loop % SystemRoot "\Microsoft.NET\Framework" (A_PtrSize=8?"64":"") "\*", 2
      if (FileExist(A_LoopFileFullPath "\mscorlib.dll") && A_LoopFileName > Version)
        Version := A_LoopFileName
  if DllCall("mscoree\CorBindToRuntimeEx", "wstr", Version, "ptr", 0, "uint", 0
  , "ptr", CLR_GUID(CLSID_CorRuntimeHost, "{CB2F6723-AB3A-11D2-9C40-00C04FA30A3E}")
  , "ptr", CLR_GUID(IID_ICorRuntimeHost,  "{CB2F6722-AB3A-11D2-9C40-00C04FA30A3E}")
  , "ptr*", RtHst) >= 0
    DllCall(NumGet(NumGet(RtHst+0)+10*A_PtrSize), "ptr", RtHst) ; Start
  return RtHst
}

;
; INTERNAL FUNCTIONS
;

CLR_GetDefaultDomain()
{
  static defaultDomain := 0
  if !defaultDomain
  {	; ICorRuntimeHost::GetDefaultDomain
    if DllCall(NumGet(NumGet(0+RtHst:=CLR_Start())+13*A_PtrSize), "ptr", RtHst, "ptr*", p:=0) >= 0
      defaultDomain := ComObject(p), ObjRelease(p)
  }
  return defaultDomain
}

CLR_CompileAssembly(Code, References, ProviderAssembly, ProviderType, AppDomain=0, FileName="", CompilerOptions="")
{
  if !AppDomain
    AppDomain := CLR_GetDefaultDomain()
  
  if !(asmProvider := CLR_LoadLibrary(ProviderAssembly, AppDomain))
  || !(codeProvider := asmProvider.CreateInstance(ProviderType))
  || !(codeCompiler := codeProvider.CreateCompiler())
    return 0

  if !(asmSystem := (ProviderAssembly="System") ? asmProvider : CLR_LoadLibrary("System", AppDomain))
    return 0
  
  ; Convert | delimited list of references into an array.
  StringSplit, Refs, References, |, %A_Space%%A_Tab%
  aRefs := ComObjArray(8, Refs0)
  Loop % Refs0
    aRefs[A_Index-1] := Refs%A_Index%
  
  ; Set parameters for compiler.
  prms := CLR_CreateObject(asmSystem, "System.CodeDom.Compiler.CompilerParameters", aRefs)
  , prms.OutputAssembly          := FileName
  , prms.GenerateInMemory        := FileName=""
  , prms.GenerateExecutable      := SubStr(FileName,-3)=".exe"
  , prms.CompilerOptions         := CompilerOptions
  , prms.IncludeDebugInformation := true
  
  ; Compile!
  compilerRes := codeCompiler.CompileAssemblyFromSource(prms, Code)
  
  if error_count := (errors := compilerRes.Errors).Count
  {
    error_text := ""
    Loop % error_count
      error_text .= ((e := errors.Item[A_Index-1]).IsWarning ? "Warning " : "Error ") . e.ErrorNumber " on line " e.Line ": " e.ErrorText "`n`n"
    MsgBox, 16, Compilation Failed, %error_text%
    return 0
  }
  ; Success. Return Assembly object or path.
  return compilerRes[FileName="" ? "CompiledAssembly" : "PathToAssembly"]
}

CLR_GUID(ByRef GUID, sGUID)
{
  VarSetCapacity(GUID, 16, 0)
  return DllCall("ole32\CLSIDFromString", "wstr", sGUID, "ptr", &GUID) >= 0 ? &GUID : ""
}

 

声明:站内资源为整理优化好的代码上传分享与学习研究,如果是开源代码基本都会标明出处,方便大家扩展学习路径。请不要恶意搬运,破坏站长辛苦整理维护的劳动成果。本站为爱好者分享站点,所有内容不作为商业行为。如若本站上传内容侵犯了原著者的合法权益,请联系我们进行删除下架。