Помогите с API!!!

Программирование на Visual Basic, главный форум. Обсуждение тем программирования на VB 1—6.
Даже если вы плохо разбираетесь в VB и программировании вообще — тут вам помогут. В разумных пределах, конечно.
Правила форума
Темы, в которых будет сначала написано «что нужно сделать», а затем просьба «помогите», будут закрыты.
Читайте требования к создаваемым темам.
smartlink
Новичок
Новичок
 
Сообщения: 44
Зарегистрирован: 22.01.2004 (Чт) 6:51
Откуда: Tashkent

Помогите с API!!!

Сообщение smartlink » 13.05.2004 (Чт) 10:13

Я нашел пример кода изменяющий параметры принтера. В данной случаи изменяет ориентацию листа с книжней на альбомную. Но я не смог добиться результата. Может посмотрите и найдете ошибку. У меня ОС Win2000 в домейне, а принтер находится на другом компе, как сетевой. Вот код и надеюсь админы форума не будут против :? :

Код: Выделить всё
Type DIVT
quot As Long
remain As Long
End Type

Private Type PRINTER_DEFAULTS
pDatatype As Long
pDevmode As Long
DesiredAccess As Long
End Type

Private Type PRINTER_INFO_2
pServerName As Long
pPrinterName As Long
pShareName As Long
pPortName As Long
pDriverName As Long
pComment As Long
pLocation As Long
pDevmode As Long
pSepFile As Long
pPrintProcessor As Long
pDatatype As Long
pParameters As Long
pSecurityDescriptor As Long
Attributes As Long
Priority As Long
DefaultPriority As Long
StartTime As Long
UntilTime As Long
Status As Long
cJobs As Long
AveragePPM As Long
End Type

Type DEVMODE
dmDeviceName As String * 32
dmSpecVersion As Integer
dmDriverVersion As Integer
dmSize As Integer
dmDriverExtra As Integer
dmFields As Long
dmOrientation As Integer
dmPaperSize As Integer
dmPaperLength As Integer
dmPaperWidth As Integer
dmScale As Integer
dmCopies As Integer
dmDefaultSource As Integer
dmPrintQuality As Integer
dmColor As Integer
dmDuplex As Integer
dmYResolution As Integer
dmTTOption As Integer
dmCollate As Integer
dmFormName As String * 32
dmUnusedPadding As Integer
dmBitsPerPel As Integer
dmPelsWidth As Long
dmPelsHeight As Long
dmDisplayFlags As Long
dmDisplayFrequency As Long
dmICMMethod As Long
dmICMIntent As Long
dmMediaType As Long
dmDitherType As Long
dmReserved1 As Long
dmReserved2 As Long
End Type

Declare Function GetProfileString Lib "Kernel32" Alias "GetProfileStringA"
(ByVal lpAppName As String, ByVal lpKeyName As String, _
ByVal lpDefault As String, ByVal lpReturnedString As String, ByVal nSize As
Long) As Long

Declare Function ClosePrinter Lib "winspool.drv" (ByVal hPrinter As Long) As
Long

Declare Function DocumentProperties Lib "winspool.drv" _
Alias "DocumentPropertiesA" (ByVal hwnd As Long, _
ByVal hPrinter As Long, ByVal pDeviceName As String, _
pDevModeOutput As Long, pDevModeInput As Long, _
ByVal fMode As Long) As Long

Declare Function GetPrinter Lib "winspool.drv" Alias _
"GetPrinterA" (ByVal hPrinter As Long, ByVal Level As Long, _
pPrinter As Long, ByVal cbBuf As Long, pcbNeeded As Long) As Long

Declare Function OpenPrinter Lib "winspool.drv" Alias _
"OpenPrinterA" (ByVal pPrinterName As String, phPrinter As Long, _
pDefault As PRINTER_DEFAULTS) As Long

Declare Function SetPrinter Lib "winspool.drv" Alias "SetPrinterA" (ByVal
hPrinter As Long, _
ByVal Level As Long, pPrinter As Long, ByVal commandl As Long) As Long

Declare Sub CopyMemory Lib "Kernel32" Alias "RtlMoveMemory" _
(pDest As DEVMODE, pSource As Long, ByVal cbLength As Long)

Declare Sub CopyMemoryBack Lib "Kernel32" Alias "RtlMoveMemory" _
(pDest As Long, pSource As DEVMODE, ByVal cbLength As Long)

Declare Sub CopyMemoryPI Lib "Kernel32" Alias "RtlMoveMemory" _
(pDest As PRINTER_INFO_2, pSource As Long, ByVal cbLength As Long)

Declare Sub CopyMemoryBackPI Lib "Kernel32" Alias "RtlMoveMemory" _
(pDest As Long, pSource As PRINTER_INFO_2, ByVal cbLength As Long)

Declare Function labs Lib "msvcrt.dll" _
(param As Long) As Long

Declare Function ldiv Lib "msvcrt.dll" (ByVal q As Long, divisor As Long) As
Long

'In the SetPrinterOrientation function:
Public Function SetPrinterOrientation(ByVal sPrinterName As String, ByVal
orientation As Integer) As Long

Const DM_IN_BUFFER = 8
Const DM_OUT_BUFFER = 2
Const PRINTER_ACCESS_ADMINISTER = &H4
Const PRINTER_ACCESS_USE = &H8
Const STANDARD_RIGHTS_REQUIRED = &HF0000
Const PRINTER_ALL_ACCESS = (STANDARD_RIGHTS_REQUIRED Or _
PRINTER_ACCESS_ADMINISTER Or PRINTER_ACCESS_USE)


Dim hPrinter As Long
Dim pd As PRINTER_DEFAULTS
Dim pinfo As PRINTER_INFO_2
Dim dm As DEVMODE
Dim dm2 As DEVMODE
Dim yDevModeData() As Long
Dim yPInfoMemory() As Long
Dim nBytesNeeded As Long
Dim nRet As Long, nJunk As Long
Dim q As Long

On Error GoTo cleanup

'Open printer with all access to be able to modify settings
pd.DesiredAccess = PRINTER_ALL_ACCESS
nRet = OpenPrinter(sPrinterName, hPrinter, pd)
nRet = DocumentProperties(0, hPrinter, sPrinterName, 0, 0, 0)
If (nRet < 0) Then
  MsgBox "Cannot get the size of the DEVMODE structure."
  GoTo cleanup
End If

ReDim yDevModeData(nRet + 100) As Long

nRet = DocumentProperties(0, hPrinter, sPrinterName, yDevModeData(0), 0,
DM_OUT_BUFFER)
If (nRet < 0) Then
  MsgBox "Cannot get the DEVMODE structure."
  GoTo cleanup
End If

Call CopyMemory(dm, yDevModeData(0), Len(dm))
dm.dmOrientation = orientation
Call CopyMemoryBack(yDevModeData(0), dm, Len(dm))
nRet = DocumentProperties(0, hPrinter, sPrinterName, _
yDevModeData(0), yDevModeData(0), _
DM_IN_BUFFER Or DM_OUT_BUFFER)

If (nRet < 0) Then
  MsgBox "Unable to set some settings to this printer."
  GoTo cleanup
End If

Call GetPrinter(hPrinter, 2, 0, 0, nBytesNeeded)
If (nBytesNeeded = 0) Then GoTo cleanup

ReDim yPInfoMemory(nBytesNeeded + 100) As Long

nRet = GetPrinter(hPrinter, 2, yPInfoMemory(0), nBytesNeeded, nJunk)
If (nRet = 0) Then
  MsgBox "Unable to get shared printer settings."
  GoTo cleanup
End If

Call CopyMemoryPI(pinfo, yPInfoMemory(0), Len(pinfo))
pinfo.pDevmode = labs(yDevModeData(0))
pinfo.pSecurityDescriptor = 0
Call CopyMemoryBackPI(yPInfoMemory(0), pinfo, Len(pinfo))

nRet = SetPrinter(hPrinter, 2, yPInfoMemory(0), 0)
If (nRet = 0) Then
  MsgBox "Unable to set shared printer settings."
End If

SetPrinterOrientation = nRet

cleanup:
If (hPrinter <> 0) Then Call ClosePrinter(hPrinter)

End Function


Private Sub Command1_Click()
Dim result As Variant
Dim printerName As String
Dim nPos As String

'Get the printer name
printerName = String(128, 0)
result = GetProfileString("WINDOWS", "DEVICE", "", printerName, 127)
nPos = InStr(printerName, ",")
printerName = Left(printerName, nPos - 1)

'Set the default printer to landscape (printer name, landscape orientation)
result = SetPrinterOrientation(printerName, 2)
End Sub

Ennor
Конструктивный критик
Конструктивный критик
 
Сообщения: 2504
Зарегистрирован: 18.12.2001 (Вт) 3:58
Откуда: Калуга -> Москва

Сообщение Ennor » 13.05.2004 (Чт) 10:21

Круто. Но не проще ли было бы:
Код: Выделить всё
Printer.Orientation = vbPRORLandscape

?

GSerg
Шаман
Шаман
 
Сообщения: 14286
Зарегистрирован: 14.12.2002 (Сб) 5:25
Откуда: Магадан

Сообщение GSerg » 13.05.2004 (Чт) 10:27

Ага... Да, я тоже порой люблю кодить на VB примерно в этом стиле :)
Принтер в системе прописан? Прописан.
Коллекция Printers есть? Есть.
Set Printer = Printers(name) могём? Могём.
Printer.Orientation = чего? vbPRORLandscape!
Как только вы переберёте все варианты решения и не найдёте нужного, тут же обнаружится решение, простое и очевидное для всех, кроме вас

smartlink
Новичок
Новичок
 
Сообщения: 44
Зарегистрирован: 22.01.2004 (Чт) 6:51
Откуда: Tashkent

Сообщение smartlink » 14.05.2004 (Пт) 7:16

Ну я так и пробовал делать! Но к сожелению ругается, говорит что: Orientation Proporty is read-only! Это я хотел использовать в DataReport, так как некоторые отчеты имеют ширину, которые не вмешаются при Portrait ориентации, а надо чтоб было Landscape. Что еще можно сделать? :( :( :( :( :( :( Может есть еще какие-нибудь подходы??? :idea: :idea: :idea:

Ennor
Конструктивный критик
Конструктивный критик
 
Сообщения: 2504
Зарегистрирован: 18.12.2001 (Вт) 3:58
Откуда: Калуга -> Москва

Сообщение Ennor » 14.05.2004 (Пт) 10:19

У меня все работает, копай права доступа. Учитывая, что у тебя домен, скорее всего здесь собака и порылась (у себя-то на работе я Domain Admin, мне он все позволяет).

smartlink
Новичок
Новичок
 
Сообщения: 44
Зарегистрирован: 22.01.2004 (Чт) 6:51
Откуда: Tashkent

Сообщение smartlink » 15.05.2004 (Сб) 9:07

У меня на принтере установлен права админа, но все равно не получается. Кроме того, на этой строке:
Код: Выделить всё
pinfo.pDevmode = labs(yDevModeData(0))


ругается:
Run-time error '49'

Bad DLL calling convention


Модель принтера Canon LASER SHOT LBP-1120.

Ennor
Конструктивный критик
Конструктивный критик
 
Сообщения: 2504
Зарегистрирован: 18.12.2001 (Вт) 3:58
Откуда: Калуга -> Москва

Сообщение Ennor » 18.05.2004 (Вт) 11:53

Рантайм №49 выскакивает, если у тебя неправильно написан Declare. Смотри внимательнее.

И еще: подозреваю, дело не в твоих правах на конкретный принтер, а в твоих правах в домене вообще. Возможно, я ошибаюсь, но это единственное, что мне приходит в голову. Еще раз повторю: у меня под домен админом все работает, причем с сетевым принтером.

Konst_One
Член-корреспондент академии VBStreets
Член-корреспондент академии VBStreets
Аватара пользователя
 
Сообщения: 3041
Зарегистрирован: 09.04.2004 (Пт) 13:47
Откуда: Химки

Сообщение Konst_One » 18.05.2004 (Вт) 12:02

Такие траблы с сетевым принтером вылезают, если ты не имеешь прав на конфигурирование этого сетевого принтера. Т.е. ты не можешь изменить его дополнительные настройки. А так как в некоторых принтерах параметры страницы (ориентация, качетсво и т.д.) устанавливаются только через дополнительные настройки (специфичные в конкретном драйвере устройства), то с помощью указания константы ориентации это сделать не представляется возможным. Ты должен иметь полные права администратора на настройку этого сетевого принтера и тогда ,возможно, у тебя что-нибудь выйдет. И указывай вместо имени константы цифру:
Код: Выделить всё
Printer.Orientation =2

Можно попробовать перебрать несколько разных цифр, например: 1, 2 или 3. Для некоторых принтеров помогает.


Вернуться в Visual Basic 1–6

Кто сейчас на конференции

Сейчас этот форум просматривают: Yandex-бот и гости: 4

    TopList